hy/hy/macros.py

122 lines
3.4 KiB
Python
Raw Normal View History

2013-03-18 15:27:14 +01:00
# Copyright (c) 2013 Paul Tagliamonte <paultag@debian.org>
2013-03-08 04:52:47 +01:00
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"),
# to deal in the Software without restriction, including without limitation
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
# and/or sell copies of the Software, and to permit persons to whom the
# Software is furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
# DEALINGS IN THE SOFTWARE.
from hy.models.expression import HyExpression
2013-03-09 22:42:07 +01:00
from hy.models.string import HyString
2013-06-07 16:35:28 +02:00
from hy.models.symbol import HySymbol
2013-03-08 04:52:47 +01:00
from hy.models.list import HyList
2013-06-05 12:19:06 +02:00
from hy.models.integer import HyInteger
from hy.models.float import HyFloat
from hy.models.complex import HyComplex
from hy.models.dict import HyDict
from hy._compat import str_type
2013-03-08 04:52:47 +01:00
from collections import defaultdict
CORE_MACROS = [
"hy.core.bootstrap",
]
EXTRA_MACROS = [
"hy.core.macros",
]
_hy_macros = defaultdict(dict)
2013-03-08 04:52:47 +01:00
def macro(name):
def _(fn):
module_name = fn.__module__
if module_name.startswith("hy.core"):
module_name = None
_hy_macros[module_name][name] = fn
2013-03-08 05:04:20 +01:00
return fn
2013-03-08 04:52:47 +01:00
return _
def require(source_module_name, target_module_name):
macros = _hy_macros[source_module_name]
refs = _hy_macros[target_module_name]
for name, macro in macros.items():
refs[name] = macro
2013-06-05 12:19:06 +02:00
def _wrap_value(x):
wrapper = _wrappers.get(type(x))
if wrapper is None:
return x
else:
return wrapper(x)
2013-06-09 02:10:27 +02:00
_wrappers = {
int: HyInteger,
bool: lambda x: HySymbol("True") if x else HySymbol("False"),
float: HyFloat,
complex: HyComplex,
str_type: HyString,
dict: lambda d: HyDict(_wrap_value(x) for x in sum(d.items(), ())),
list: lambda l: HyList(_wrap_value(x) for x in l)
}
2013-06-05 12:19:06 +02:00
def process(tree, module_name):
load_macros(module_name)
old = None
while old != tree:
old = tree
tree = macroexpand(tree, module_name)
return tree
def load_macros(module_name):
for module in CORE_MACROS:
__import__(module)
if module_name.startswith("hy.core"):
return
for module in EXTRA_MACROS:
__import__(module)
def macroexpand(tree, module_name):
2013-03-09 00:46:51 +01:00
if isinstance(tree, HyExpression):
if tree == []:
return tree
2013-06-26 08:44:09 +02:00
fn = tree[0]
2013-05-11 00:29:42 +02:00
if fn in ("quote", "quasiquote"):
return tree
ntree = HyExpression(tree[:])
2013-03-14 01:41:53 +01:00
ntree.replace(tree)
2013-03-09 00:46:51 +01:00
2013-03-09 22:42:07 +01:00
if isinstance(fn, HyString):
m = _hy_macros[module_name].get(fn)
if m is None:
m = _hy_macros[None].get(fn)
if m is not None:
2013-06-05 12:19:06 +02:00
obj = _wrap_value(m(*ntree[1:]))
2013-03-09 22:42:07 +01:00
obj.replace(tree)
return obj
2013-03-09 00:46:51 +01:00
return ntree
2013-03-08 04:52:47 +01:00
return tree