blob: e3685eeae3a48e354b7338a24a72009cfb0cdc36 [file] [log] [blame]
Christopher Piro094823a2007-07-18 00:26:12 +00001%%% 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 Piro5b3a8f72007-08-01 22:27:37 +00009-export([get/2, set/3, call/2, call/3, inspect/1, start_new/2, is_object/1, class/1]).
Christopher Piro094823a2007-07-18 00:26:12 +000010-export([behaviour_info/1]).
11
Christopher Piro5b3a8f72007-08-01 22:27:37 +000012-include("thrift.hrl").
Christopher Piro094823a2007-07-18 00:26:12 +000013-include("oop.hrl").
14
15%%%
16%%% behavior definition
17%%%
18
19behaviour_info(callbacks) ->
20 [
21 {attr, 4},
22 {super, 0}
23 ];
24behaviour_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
45get(Obj, Field) ->
46 call(Obj, attr, [get, Field, get]).
47
48set(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
73get_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 Piro5b3a8f72007-08-01 22:27:37 +000081is_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;
89is_object(_) ->
90 false.
91
Christopher Piro094823a2007-07-18 00:26:12 +000092call(Obj, Method, ArgsProper) ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +000093 %% error_logger:info_msg("call called: Obj=~p Method=~p ArgsProper=~p", [inspect(Obj), Method, ArgsProper]),
Christopher Piro094823a2007-07-18 00:26:12 +000094 Args = [Obj|ArgsProper], %% prepend This to args
95 TryModule = ?CLASS(Obj),
96 call_loop(Obj, Method, Args, TryModule, [], Obj).
97
98call(Obj, Method) ->
99 call(Obj, Method, []).
100
101call_loop(Obj, Method, Args, TryModule, TriedRev, FirstObj) ->
102 try
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000103 %% error_logger:info_msg("call_loop~n ~p~n ~p~n ~p~n ~p", [inspect(Obj), Method, Args, TryModule]),
Christopher Piro094823a2007-07-18 00:26:12 +0000104 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 Piro5b3a8f72007-08-01 22:27:37 +0000136class(Obj) when is_tuple(Obj) ->
137 case is_object(Obj) of
138 true ->
139 ?CLASS(Obj);
140 false ->
141 none
142 end;
143class(_) ->
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 Piro094823a2007-07-18 00:26:12 +0000149inspect(Obj) ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000150 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 Piro094823a2007-07-18 00:26:12 +0000165
166inspect_loop(Obj, Str) ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000167 Class = ?CLASS(Obj),
168 Inspect = Class:inspect(Obj),
169 Current = atom_to_list(Class) ++ ": " ++ Inspect,
170
Christopher Piro094823a2007-07-18 00:26:12 +0000171 case get_superobject(Obj) of
172 { ok, Superobj } ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000173 inspect_loop(Superobj, Str ++ Current ++ " | ");
Christopher Piro094823a2007-07-18 00:26:12 +0000174 none ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000175 Str ++ Current ++ ">"
Christopher Piro094823a2007-07-18 00:26:12 +0000176 end.
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000177
Christopher Piro094823a2007-07-18 00:26:12 +0000178%% TODO: voids take only ok as return?
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000179start_new(none=Resv, _) ->
180 error_logger:format("can't instantiate ~p: class name is a reserved word", [Resv]),
181 error;
Christopher Piro094823a2007-07-18 00:26:12 +0000182start_new(Class, Args) ->
Christopher Piro5b3a8f72007-08-01 22:27:37 +0000183 {ok, Pid} = gen_server:start_link(thrift_oop_server, {Class, Args}, []),
184 Pid.