johnbender / unraverl

An erlang module built to provide language features and other operations through manipulation of the abstract form.

This URL has Read+Write access

johnbender (author)
Sat Aug 08 18:38:57 -0700 2009
commit  cc8310fab4538cefd035502f95327bf894130292
tree    6c650a303b615b99e0220b92d9bc8459be6dbc10
parent  f66c0ed3801b3aa8aa34eaaefee06f06086f0744
unraverl / filters.erl
100644 95 lines (74 sloc) 3.34 kb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
-module(filters).
-export([before_exec/3, after_exec/3]).
 
-define(before_form, {function,LineNum,Original,Arity,
[{clause,
LineNum,
OriginalArgs,
[],
[{match,LineNum,FilterResultArgs,
{call,LineNum ,{atom, LineNum, Filter},FilterArgs}},
{call,LineNum,
{atom,LineNum, RenamedFunc},
RenamedFuncArgs}]}]}).
 
-define(after_form, {function,LineNum,Original,Arity,
[{clause, LineNum,
OriginalArgs,
[],
[{match,LineNum,
RenamdFuncResultArg,
{call,LineNum,{atom,LineNum, RenamedFunc}, OriginalArgs}},
{call,LineNum,{atom,LineNum,Filter},FilterArgs}]}]}).
 
 
before_exec(Form, Funcs, Filter) ->
    Fun = fun(A, B, C, D) -> before_exec_function(A, B, C, D) end,
    apply_alteration(Form, Funcs, Filter, Fun).
 
after_exec(Form, Funcs, Filter) ->
    Fun = fun(A, B, C, D) -> after_exec_function(A, B, C, D) end,
    apply_alteration(Form, Funcs, Filter, Fun).
 
apply_alteration(Form, [], _filter, _alteration) -> Form;
apply_alteration(Form, [Original|T], Filter, FunctionAddition) ->
    {function, LineNum, Original, Arity, Clauses} = util:find_function(Form, Original),
    {function, _linenum, Filter, FilterArity, _clauses} = util:find_function(Form, Filter),
    RenamedFunc = rename(before, Original, Filter),
    Renamed = {function, LineNum, RenamedFunc, Arity, Clauses},
    Replaced = lists:keyreplace(Original, 3, Form, Renamed),
    Appended = append(Replaced, FunctionAddition({Original, Arity}, {Filter, FilterArity}, RenamedFunc, eof_line(Replaced))),
    apply_alteration(Appended, T, Filter, FunctionAddition).
    
 
before_exec_function({Original, Arity}, {Filter, FArity}, RenamedFunc, LineNum) ->
    OriginalArgs = util:args_list_form(Arity, LineNum, "Original"),
    FilterResultArgs = var_list_form(Arity, LineNum, "Result"),
    FilterArgs = case {Arity == FArity, FArity == 1} of
{true, _} ->
util:args_list_form(Arity, LineNum, "Original");
{_, true} ->
[var_list_form(Arity, LineNum, "Original")];
_ -> throw(invalide_filter_arity)
end,
    RenamedFuncArgs = util:args_list_form(Arity, LineNum, "Result"),
    ?before_form.
 
after_exec_function({Original, Arity}, {Filter, FArity}, RenamedFunc, LineNum) ->
    OriginalArgs = util:args_list_form(Arity, LineNum, "Original"),
    RenamdFuncResultArg = var_form(LineNum, "Result1"),
    FilterArgs = case FArity == 1 of
true ->
util:args_list_form(1, LineNum, "Result");
false -> throw(invalide_filter_arity)
end,
    ?after_form.
    
var_list_form(0, LineNum, _nameseed) ->
    {nil, LineNum};
 
var_list_form(Count, LineNum, NameSeed) ->
    VarName = list_to_atom(NameSeed ++ util:to_string(Count)),
    {cons, LineNum, var_form(LineNum, VarName), var_list_form(Count-1, LineNum, NameSeed)}.
    
    
var_form(Num, Name) when is_list(Name) ->
    var_form(Num, list_to_atom(Name));
 
var_form(Num, Name) when is_number(Num) ->
    {var, Num, Name}.
 
rename(Type, Original, Filter) ->
    Name = lists:flatten(io_lib:format("~s_~s_~s" , [util:to_string(Type), util:to_string(Original), util:to_string(Filter)])),
    list_to_atom(Name).
 
 
append(Form, Addition) ->
    End = {eof, _} = lists:last(Form),
    lists:delete(End, Form) ++ [Addition, End].
    
eof_line(Form)->
    {eof, LineNum} = lists:last(Form),
    LineNum.