/usr/lib/swi-prolog/boot/dicts.pl is in swi-prolog-nox 7.2.3-2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
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 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 | :- module('$dicts',
[ '.'/3 % +Left, +Right, -Result
]).
%% .(+R, +L, -Result)
%
% Evaluate dot expressions. Note that '$get_dict_ex' fails if the
% first argument is not a dict or the second is not a valid key or
% unbound.
.(Data, Func, Value) :-
( '$get_dict_ex'(Func, Data, V0)
*-> Value = V0
; is_dict(Data, Tag)
-> eval_dict_function(Func, Tag, Data, Value)
; is_list(Data)
-> ( (atomic(Func) ; var(Func))
-> dict_create(Dict, _, Data),
'$get_dict_ex'(Func, Dict, Value)
; '$type_error'(atom, Func)
)
; '$type_error'(dict, Data)
).
%% eval_dict_function(+Func, +Tag, +Dict, -Value)
%
% Test for predefined functions on dicts or evaluate a user-defined
% function.
eval_dict_function(get(Key), _, Dict, Value) :- !,
get_dict(Key, Dict, Value).
eval_dict_function(put(Key, Value), _, Dict, NewDict) :- !,
( atomic(Key)
-> put_dict(Key, Dict, Value, NewDict)
; put_dict_path(Key, Dict, Value, NewDict)
).
eval_dict_function(put(New), _, Dict, NewDict) :- !,
put_dict(New, Dict, NewDict).
eval_dict_function(Func, Tag, Dict, Value) :-
call(Tag:Func, Dict, Value).
%% put_dict_path(+KeyPath, +Dict, +Value, -NewDict)
%
% Add/replace a value according to a path definition. Path
% segments are separated using '/'.
put_dict_path(Key, Dict, Value, NewDict) :-
atom(Key), !,
put_dict(Key, Dict, Value, NewDict).
put_dict_path(Path, Dict, Value, NewDict) :-
get_dict_path(Path, Dict, _Old, NewDict, Value).
get_dict_path(Path, _, _, _, _) :-
var(Path), !,
'$instantiation_error'(Path).
get_dict_path(Path/Key, Dict, Old, NewDict, New) :- !,
get_dict_path(Path, Dict, OldD, NewDict, NewD),
( get_dict(Key, OldD, Old, NewD, New),
is_dict(Old)
-> true
; Old = _{},
put_dict(Key, OldD, New, NewD)
).
get_dict_path(Key, Dict, Old, NewDict, New) :-
get_dict(Key, Dict, Old, NewDict, New),
is_dict(Old), !.
get_dict_path(Key, Dict, _{}, NewDict, New) :-
put_dict(Key, Dict, New, NewDict).
/*******************************
* REGISTER *
*******************************/
%% system:term_expansion(+TermIn, -TermOut)
%
% Support := syntax for defining new functions.
%
% @tbd Modify to term_expansion/4, including position
% management
% note that we need FHead because using a term there rewrites the
% clauses using function expansion.
expand_dict_function((QFHead := V0 :- Body), (QHead :- Body, Eval)) :-
fqhead(QFHead, FHead, Head, QHead),
FHead =.. [.,R,M], !,
'$expand':replace_functions(V0, Eval, V, _Ctx),
compound_name_arguments(M, Name, Args0),
'$append'(Args0, [R,V], Args),
compound_name_arguments(Head, Name, Args).
expand_dict_function((QFHead := V0), (QHead :- Eval)) :-
fqhead(QFHead, FHead, Head, QHead),
FHead =.. [.,R,M], !,
'$expand':replace_functions(V0, Eval, V, _Ctx),
compound_name_arguments(M, Name, Args0),
'$append'(Args0, [R,V], Args),
compound_name_arguments(Head, Name, Args).
fqhead(M:FHead, FHead, Head, M:Head) :- !.
fqhead(FHead, FHead, Head, Head).
system:term_expansion(FDecl, Clause) :-
expand_dict_function(FDecl, Clause).
system:term_expansion(M:FDecl, QClause) :-
expand_dict_function(FDecl, Clause), !,
QClause = M:Clause.
|