Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 1 | %%% Copyright (c) 2007- Facebook |
| 2 | %%% Distributed under the Thrift Software License |
| 3 | %%% |
| 4 | %%% See accompanying file LICENSE or visit the Thrift site at: |
| 5 | %%% http://developers.facebook.com/thrift/ |
| 6 | |
| 7 | -module(oop). |
| 8 | |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 9 | -export([get/2, set/3, call/2, call/3, inspect/1, start_new/2, is_object/1, class/1]). |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 10 | -export([behaviour_info/1]). |
| 11 | |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 12 | -include("thrift.hrl"). |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 13 | -include("oop.hrl"). |
| 14 | |
| 15 | %%% |
| 16 | %%% behavior definition |
| 17 | %%% |
| 18 | |
| 19 | behaviour_info(callbacks) -> |
| 20 | [ |
| 21 | {attr, 4}, |
| 22 | {super, 0} |
| 23 | ]; |
| 24 | behaviour_info(_) -> |
| 25 | undefined. |
| 26 | |
| 27 | %% |
| 28 | |
| 29 | -define(TRIED, lists:reverse([TryModule|TriedRev])). |
| 30 | |
| 31 | %% no super attr defined |
| 32 | -define(NOSUPEROBJ, exit({missing_attr_super, {inspect(Obj), ?TRIED}})). |
| 33 | |
| 34 | -define(NOMETHOD, exit({missing_method, {Method, inspect(Obj), tl(Args), ?TRIED}})). |
| 35 | |
| 36 | -define(NOATTR, exit({missing_attr, {hd(tl(Args)), inspect(FirstObj), ?TRIED}})). |
| 37 | |
| 38 | -define(NOATTR_SET, exit({missing_attr, {Field, inspect(Obj), ".." %% TODO: give a backtrace |
| 39 | }})). |
| 40 | |
| 41 | |
| 42 | %%% get(Obj, Field) -> term() |
| 43 | %%% looks up Field in Obj or its ancestor objects |
| 44 | |
| 45 | get(Obj, Field) -> |
| 46 | call(Obj, attr, [get, Field, get]). |
| 47 | |
| 48 | set(Obj, Field, Value) -> %% TODO: could be tail-recursive |
| 49 | Module = ?CLASS(Obj), |
| 50 | try |
| 51 | Module:attr(Obj, set, Field, Value) |
| 52 | catch |
| 53 | error:Kind when Kind == undef; Kind == function_clause -> |
| 54 | case get_superobject(Obj) of |
| 55 | { ok, Superobj } -> |
| 56 | Super1 = set(Superobj, Field, Value), |
| 57 | try |
| 58 | Module:attr(Obj, set, super, Super1) |
| 59 | catch %% TODO(cpiro): remove check |
| 60 | X -> exit({burnsauce, X}) |
| 61 | end; |
| 62 | none -> |
| 63 | ?NOATTR_SET |
| 64 | end |
| 65 | end. |
| 66 | |
| 67 | |
| 68 | %%% C++ <-> Erlang |
| 69 | %%% classes modules |
| 70 | %%% class b : public a a:super() -> b. |
| 71 | %%% |
| 72 | |
| 73 | get_superobject(Obj) -> |
| 74 | try |
| 75 | {ok, (?CLASS(Obj)):attr(Obj, get, super, get)} |
| 76 | catch |
| 77 | error:Kind when Kind == undef; Kind == function_clause -> |
| 78 | none |
| 79 | end. |
| 80 | |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 81 | is_object(Obj) when is_tuple(Obj) -> |
| 82 | try |
| 83 | (?CLASS(Obj)):super(), %% if it's an object its first element will be a class name, and it'll have super/0 |
| 84 | true |
| 85 | catch |
| 86 | error:Kind when Kind == undef; Kind == function_clause -> |
| 87 | false |
| 88 | end; |
| 89 | is_object(_) -> |
| 90 | false. |
| 91 | |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 92 | call(Obj, Method, ArgsProper) -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 93 | %% error_logger:info_msg("call called: Obj=~p Method=~p ArgsProper=~p", [inspect(Obj), Method, ArgsProper]), |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 94 | Args = [Obj|ArgsProper], %% prepend This to args |
| 95 | TryModule = ?CLASS(Obj), |
| 96 | call_loop(Obj, Method, Args, TryModule, [], Obj). |
| 97 | |
| 98 | call(Obj, Method) -> |
| 99 | call(Obj, Method, []). |
| 100 | |
| 101 | call_loop(Obj, Method, Args, TryModule, TriedRev, FirstObj) -> |
| 102 | try |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 103 | %% error_logger:info_msg("call_loop~n ~p~n ~p~n ~p~n ~p", [inspect(Obj), Method, Args, TryModule]), |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 104 | apply(TryModule, Method, Args) |
| 105 | catch |
| 106 | error:Kind when Kind == undef; Kind == function_clause -> |
| 107 | case { TryModule:super(), Method } of |
| 108 | { none, attr } -> |
| 109 | ?NOATTR; |
| 110 | |
| 111 | { none, _ } -> |
| 112 | ?NOMETHOD; |
| 113 | |
| 114 | { Superclass, attr } -> |
| 115 | %% look for attrs in the "super object" |
| 116 | |
| 117 | case get_superobject(Obj) of |
| 118 | {ok, Superobj} when (TryModule == ?CLASS(Obj)) -> |
| 119 | %% replace This with Superobj |
| 120 | NewArgs = [Superobj|tl(Args)], |
| 121 | call_loop(Superobj, Method, NewArgs, |
| 122 | Superclass, [TryModule|TriedRev], FirstObj); |
| 123 | |
| 124 | {ok, _Superobj} -> % failed guard TODO(cpiro): removeme |
| 125 | exit(oh_noes); |
| 126 | |
| 127 | none -> ?NOSUPEROBJ |
| 128 | end; |
| 129 | |
| 130 | { SuperClass, _ } -> |
| 131 | call_loop(Obj, Method, Args, |
| 132 | SuperClass, [TryModule|TriedRev], FirstObj) |
| 133 | end |
| 134 | end. |
| 135 | |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 136 | class(Obj) when is_tuple(Obj) -> |
| 137 | case is_object(Obj) of |
| 138 | true -> |
| 139 | ?CLASS(Obj); |
| 140 | false -> |
| 141 | none |
| 142 | end; |
| 143 | class(_) -> |
| 144 | none. |
| 145 | |
| 146 | %% careful: not robust against records beginning with a class name |
| 147 | %% (note: we can't just guard with is_record(?CLASS(Obj), Obj) since we |
| 148 | %% can't/really really shouldn't require all record definitions in this file |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 149 | inspect(Obj) -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 150 | try |
| 151 | case is_object(Obj) of |
| 152 | true -> |
| 153 | DeepList = inspect_loop(Obj, "#<"), |
| 154 | lists:flatten(DeepList); |
| 155 | false -> |
| 156 | thrift_utils:sformat("~p", [Obj]) |
| 157 | end |
| 158 | catch |
| 159 | _:E -> |
| 160 | thrift_utils:sformat("INSPECT_ERROR(~p) ~p", [E, Obj]) |
| 161 | |
| 162 | %% TODO(cpiro): bring this back once we're done testing: |
| 163 | %% _:E -> thrift_utils:sformat("~p", [Obj]) |
| 164 | end. |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 165 | |
| 166 | inspect_loop(Obj, Str) -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 167 | Class = ?CLASS(Obj), |
| 168 | Inspect = Class:inspect(Obj), |
| 169 | Current = atom_to_list(Class) ++ ": " ++ Inspect, |
| 170 | |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 171 | case get_superobject(Obj) of |
| 172 | { ok, Superobj } -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 173 | inspect_loop(Superobj, Str ++ Current ++ " | "); |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 174 | none -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 175 | Str ++ Current ++ ">" |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 176 | end. |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 177 | |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 178 | %% TODO: voids take only ok as return? |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 179 | start_new(none=Resv, _) -> |
| 180 | error_logger:format("can't instantiate ~p: class name is a reserved word", [Resv]), |
| 181 | error; |
Christopher Piro | 094823a | 2007-07-18 00:26:12 +0000 | [diff] [blame] | 182 | start_new(Class, Args) -> |
Christopher Piro | 5b3a8f7 | 2007-08-01 22:27:37 +0000 | [diff] [blame] | 183 | {ok, Pid} = gen_server:start_link(thrift_oop_server, {Class, Args}, []), |
| 184 | Pid. |