aboutsummaryrefslogblamecommitdiffstats
path: root/lib/diameter/src/compiler/diameter_make.erl
blob: 0d8cdec7f295f72930663eb125960f77a9dbbc13 (plain) (tree)
1
2
3
4


                   
                                                        















                                                                         

                                                              
                                            
  
                       
                                
                                                             
                         



                       

                 
                

                  
                     
 

                      
                             
                                                              
                     

                      
 






                                                                



                                                                              



                                                                    
 
                          
        
               
                     

                                                                          
                         
 
                    


                                            


                         
 

                    
 




                       
         


                                                                     
 
                         
                             

                       
 





                               
        
 

                   
 



























                                                                        

           
                                                               






                                    
            
  
                                               
 

                                
 













                                                               
 
                                                                              
 










                                                                             
       



                               
       
                                                          

                        
                                                                 
        
%%
%% %CopyrightBegin%
%%
%% Copyright Ericsson AB 2010-2013. All Rights Reserved.
%%
%% The contents of this file are subject to the Erlang Public License,
%% Version 1.1, (the "License"); you may not use this file except in
%% compliance with the License. You should have received a copy of the
%% Erlang Public License along with this software. If not, it can be
%% retrieved online at http://www.erlang.org/.
%%
%% Software distributed under the License is distributed on an "AS IS"
%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
%% the License for the specific language governing rights and limitations
%% under the License.
%%
%% %CopyrightEnd%
%%

%%
%% Module alternative to diameterc for dictionary compilation.
%%
%% Eg. 1> diameter_make:codec("mydict.dia").
%%
%%     $ erl -noinput \
%%           -boot start_clean \
%%           -eval 'ok = diameter_make:codec("mydict.dia")' \
%%           -s init stop
%%

-module(diameter_make).

-export([codec/1,
         codec/2,
         dict/1,
         dict/2,
         format/1,
         flatten/1]).

-export_type([opt/0]).

%% Options passed to codec/2.
-type opt() :: {include|outdir|name|prefix|inherits, string()}
             | return
             | verbose
             | debug.

%% Literal dictionary or path. A NL of CR identifies the former.
-type dict() :: iolist()
              | binary().

%% Name of a literal dictionary if otherwise unspecified.
-define(DEFAULT_DICT_NAME, "dictionary.dia").

%% ===========================================================================

%% codec/1-2
%%
%% Parse a dictionary file and generate a codec module. Input
%% dictionary can be either a path or the dictionary itself: the
%% occurrence of \n or \r in the argument is used to distinguish the
%% two.

-spec codec(File, [opt()])
   -> ok
    | {ok, Ret}
    | {error, Reason}
 when File :: dict(),
      Ret  :: list(),  %% [Erl, Hrl | Debug], Debug = [] | [ParseD, Forms]
      Reason :: string().

codec(File, Opts) ->
    case to_dict(File, Opts) of
        {ok, {Dict, Dictish}} ->
            make(file(Dictish), Opts, Dict);
        {error, _} = E ->
            E
    end.

codec(File) ->
    codec(File, []).

file({path, Path}) ->
    Path;
file(_) ->
    ?DEFAULT_DICT_NAME.

%% dict/2
%%
%% Parse a dictionary file and return the orddict that a codec module
%% returns from dict/0.

-spec dict(File, [opt()])
   -> {ok, orddict:orddict()}
    | {error, string()}
  when File :: dict().

dict(File, Opts) ->
    case to_dict(File, Opts) of
        {ok, {Dict, _}} ->
            {ok, Dict};
        {error, _} = E ->
            E
    end.

dict(File) ->
    dict(File, []).

%% to_dict/2

to_dict(File, Opts) ->
    Dictish = maybe_path(File),
    case diameter_dict_util:parse(Dictish, Opts) of
        {ok, Dict} ->
            {ok, {Dict, Dictish}};
        {error = E, Reason} ->
            {E, diameter_dict_util:format_error(Reason)}
    end.

maybe_path(File) ->
    Bin = iolist_to_binary([File]),
    case is_path(Bin) of
        true  -> {path, File};
        false -> Bin
    end.

%% Interpret anything containing \n or \r as a literal dictionary,
%% otherwise a path. (Which might be the wrong guess in the worst case.)
is_path(Bin) ->
    try
        [throw(C) || <<C>> <= Bin, $\n == C orelse $\r == C],
        true
    catch
        throw:_ -> false
    end.

%% format/1
%%
%% Turn an orddict returned by dict/1-2 back into a dictionary.

-spec format(orddict:orddict())
   -> iolist().

format(Dict) ->
    diameter_dict_util:format(Dict).

%% flatten/1
%%
%% Reconstitute a dictionary without @inherits.

-spec flatten(orddict:orddict())
   -> orddict:orddict().

flatten(Dict) ->
    lists:foldl(fun flatten/2, Dict, [[avp_types, import_avps],
                                      [grouped, import_groups],
                                      [enum, import_enums]]).

flatten([_,_] = Keys, Dict) ->
    [Values, Imports] = [orddict:fetch(K, Dict) || K <- Keys],
    Vs = lists:append([Values | [V || {_,V} <- Imports]]),
    lists:foldl(fun store/2,
                Dict,
                lists:zip([inherits | Keys], [[], Vs, []])).

store({Key, Value}, Dict) ->
    orddict:store(Key, Value, Dict).

%% ===========================================================================

make(File, Opts, Dict) ->
    ok(lists:foldl(fun(M,A) -> [make(File, Opts, Dict, M) | A] end,
                   [],
                   lists:append([[dict, forms] || lists:member(debug, Opts)])
                   ++ [erl, hrl])).
%% The order in which results are generated (dict/forms/erl/hrl) is
%% intentional, in order of more processing (except for hrl, which
%% isn't needed by diameter itself), since an error raises an
%% exception. The order of return results is the reverse.

ok([ok,_|_]) ->
    ok;
ok([_,_|_] = L) ->
    {ok, L}.

make(File, Opts, Dict, Mode) ->
    try
        diameter_codegen:from_dict(File, Dict, Opts, Mode)
    catch
        error: Reason ->
            erlang:error({Reason, Mode, erlang:get_stacktrace()})
    end.