Sidef.pm 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471
  1. package Sidef {
  2. use utf8;
  3. use 5.016;
  4. our $VERSION = '22.05';
  5. our $SPACES = 0; # the current number of indentation spaces
  6. our $SPACES_INCR = 4; # the number of indentation spaces
  7. our %INCLUDED; # will keep track of included modules
  8. our %EVALS; # will contain info required for eval()
  9. use constant {
  10. UPDATE_SEC => 5 * 60 * 60, # 5 hours
  11. DELETE_SEC => 2 * 24 * 60 * 60, # 2 days
  12. SANITIZE_SEC => 3 * 24 * 60 * 60, # 3 days
  13. };
  14. use List::Util qw();
  15. use File::Spec qw();
  16. use Sidef::Types::Bool::Bool;
  17. use Sidef::Types::Number::Number;
  18. sub new {
  19. my ($class, %opt) = @_;
  20. bless \%opt, $class;
  21. }
  22. *call = \&new;
  23. sub parse_code {
  24. my ($self, $code) = @_;
  25. local %INCLUDED;
  26. $self->{parser} //= Sidef::Parser->new(
  27. opt => $self->{opt},
  28. file_name => $self->{name} // '-',
  29. script_name => $self->{name} // '-',
  30. ($self->{parser_opt} ? (%{$self->{parser_opt}}) : ()),
  31. );
  32. my $ast = $self->{parser}->parse_script(code => \$code);
  33. # Check for optimization
  34. if (defined(my $level = $self->{opt}{O})) {
  35. # Optimize the AST
  36. if ($level >= 1) {
  37. $ast = $self->optimize_ast($ast);
  38. }
  39. # Deparse the AST into code, then parse the code again.
  40. if ($level >= 2) {
  41. my $sidef = Sidef->new(
  42. opt => $self->{opt},
  43. name => $self->{name},
  44. parser_opt => $self->{parser_opt},
  45. );
  46. local $sidef->{opt}{O} = 1;
  47. return $sidef->parse_code($self->compile_ast($ast, 'Sidef'));
  48. }
  49. }
  50. return $ast;
  51. }
  52. sub optimize_ast {
  53. my ($self, $ast) = @_;
  54. my $optimizer = Sidef::Optimizer->new;
  55. scalar {$optimizer->optimize($ast)};
  56. }
  57. sub execute_code {
  58. my ($self, $code) = @_;
  59. $self->execute_perl($self->compile_code($code, 'Perl'));
  60. }
  61. sub execute_perl {
  62. my ($self, $code) = @_;
  63. local $Sidef::PARSER = $self->{parser};
  64. local $Sidef::DEPARSER = $self->{Perl}{deparser};
  65. eval($code);
  66. }
  67. sub get_sidef_config_dir {
  68. my ($self) = @_;
  69. $self->{sidef_config_dir} //= $ENV{SIDEF_CONFIG_DIR}
  70. || File::Spec->catdir(
  71. $ENV{XDG_CONFIG_DIR}
  72. || (
  73. $ENV{HOME}
  74. || $ENV{LOGDIR}
  75. || (
  76. $^O eq 'MSWin32'
  77. ? '\Local Settings\Application Data'
  78. : eval { ((getpwuid($<))[7] || `echo -n ~`) }
  79. )
  80. || File::Spec->curdir()
  81. ),
  82. '.config',
  83. 'sidef'
  84. );
  85. if (not -d $self->{sidef_config_dir}) {
  86. require File::Path;
  87. eval { File::Path::make_path($self->{sidef_config_dir}) }
  88. or warn "[WARNING] Can't create directory <<$self->{sidef_config_dir}>>: $!";
  89. }
  90. return $self->{sidef_config_dir};
  91. }
  92. sub get_sidef_vdir {
  93. my ($self) = @_;
  94. $self->{_sidef_vdir} //= File::Spec->catdir($self->get_sidef_config_dir, "v$VERSION");
  95. }
  96. sub has_dbm_driver {
  97. my ($self) = @_;
  98. if (exists $self->{dbm_driver}) {
  99. return $self->{dbm_driver};
  100. }
  101. if (eval { require DB_File; 1 }) {
  102. return ($self->{dbm_driver} = 'bdbm');
  103. }
  104. if (eval { require GDBM_File; 1 }) {
  105. return ($self->{dbm_driver} = 'gdbm');
  106. }
  107. warn "No supported database driver was found...\n";
  108. warn "Please install DB_File or GDBM_File in order to use this functionality.\n";
  109. return;
  110. }
  111. sub _init_db {
  112. my ($self, $hash, $db_file) = @_;
  113. dbmopen(%$hash, $db_file, 0640);
  114. }
  115. sub _init_time_db {
  116. my ($self, $lang) = @_;
  117. if (not exists $self->{$lang}{_time_hash}) {
  118. $self->{$lang}{_time_hash} = {};
  119. $self->_init_db($self->{$lang}{_time_hash}, $self->{$lang}{time_db});
  120. if (not exists $self->{$lang}{_time_hash}{sanitized}) {
  121. $self->{$lang}{_time_hash}{sanitized} = time;
  122. }
  123. }
  124. }
  125. sub _init_code_db {
  126. my ($self, $lang) = @_;
  127. if (not exists $self->{$lang}{_code_hash}) {
  128. $self->{$lang}{_code_hash} = {};
  129. $self->_init_db($self->{$lang}{_code_hash}, $self->{$lang}{code_db});
  130. }
  131. }
  132. sub dbm_lookup {
  133. my ($self, $lang, $md5) = @_;
  134. $self->_init_time_db($lang)
  135. if not exists($self->{$lang}{_time_hash});
  136. if (exists($self->{$lang}{_time_hash}{$md5})) {
  137. $self->_init_code_db($lang)
  138. if not exists($self->{$lang}{_code_hash});
  139. if (time - $self->{$lang}{_time_hash}{$md5} >= UPDATE_SEC) {
  140. $self->{$lang}{_time_hash}{$md5} = time;
  141. }
  142. my $compressed_code = $self->{$lang}{_code_hash}{$md5};
  143. state $_x = require IO::Uncompress::RawInflate;
  144. IO::Uncompress::RawInflate::rawinflate(\$compressed_code => \my $decompressed_code)
  145. or die "rawinflate failed: $IO::Uncompress::RawInflate::RawInflateError";
  146. return Encode::decode_utf8($decompressed_code);
  147. }
  148. return;
  149. }
  150. sub dbm_store {
  151. my ($self, $lang, $md5, $code) = @_;
  152. $self->_init_code_db($lang)
  153. if not exists($self->{$lang}{_code_hash});
  154. # Sanitize the database, by removing old entries
  155. if (time - $self->{$lang}{_time_hash}{sanitized} >= SANITIZE_SEC) {
  156. $self->{$lang}{_time_hash}{sanitized} = time;
  157. my @delete_keys;
  158. while (my ($key, $value) = each %{$self->{$lang}{_time_hash}}) {
  159. if (time - $value >= DELETE_SEC) {
  160. push @delete_keys, $key;
  161. }
  162. }
  163. if (@delete_keys) {
  164. delete @{$self->{$lang}{_time_hash}}{@delete_keys};
  165. delete @{$self->{$lang}{_code_hash}}{@delete_keys};
  166. }
  167. }
  168. state $_x = require IO::Compress::RawDeflate;
  169. IO::Compress::RawDeflate::rawdeflate(\$code => \my $compressed_code)
  170. or die "rawdeflate failed: $IO::Compress::RawDeflate::RawDeflateError";
  171. $self->{$lang}{_time_hash}{$md5} = time;
  172. $self->{$lang}{_code_hash}{$md5} = $compressed_code;
  173. }
  174. sub compile_code {
  175. my ($self, $code, $lang) = @_;
  176. $lang //= 'Sidef';
  177. if (
  178. $self->{opt}{s}
  179. ##and length($$code) > 1024
  180. and $self->has_dbm_driver
  181. ) {
  182. my $db_dir = ($self->{$lang}{db_dir} //= File::Spec->catdir($self->get_sidef_vdir(), $lang));
  183. if (not -e $db_dir) {
  184. require File::Path;
  185. File::Path::make_path($db_dir);
  186. }
  187. state $_x = do {
  188. require Encode;
  189. require Digest::MD5;
  190. };
  191. my $md5 = Digest::MD5::md5_hex(Encode::encode_utf8($code));
  192. $self->{$lang}{time_db} //= File::Spec->catfile($db_dir, 'Sidef_Time_' . $self->{dbm_driver} . '.db');
  193. $self->{$lang}{code_db} //= File::Spec->catfile($db_dir, 'Sidef_Code_' . $self->{dbm_driver} . '.db');
  194. if (defined(my $cached_code = $self->dbm_lookup($lang, $md5))) {
  195. return $cached_code;
  196. }
  197. my $evals_num = keys(%EVALS);
  198. local $self->{environment_name} = 'Sidef::Runtime' . $md5;
  199. my $deparsed = $self->compile_ast($self->parse_code($code), $lang);
  200. if ($lang eq 'Perl') {
  201. $deparsed = "package $self->{environment_name} {$deparsed}\n";
  202. }
  203. # Don't store code that contains eval()
  204. if (keys(%EVALS) == $evals_num) {
  205. $self->dbm_store($lang, $md5, Encode::encode_utf8($deparsed));
  206. }
  207. return $deparsed;
  208. }
  209. state $count = 0;
  210. local $self->{environment_name} = 'Sidef::Runtime' . (CORE::abs($count++) || '');
  211. my $deparsed = $self->compile_ast($self->parse_code($code), $lang);
  212. if ($lang eq 'Perl') {
  213. $deparsed = "package $self->{environment_name} {$deparsed}\n";
  214. }
  215. return $deparsed;
  216. }
  217. sub compile_ast {
  218. my ($self, $ast, $lang) = @_;
  219. $lang //= 'Sidef';
  220. my $module = "Sidef::Deparse::$lang";
  221. my $pm = ($module =~ s{::}{/}gr . '.pm');
  222. require $pm;
  223. $self->{$lang}{deparser} = $module->new(opt => $self->{opt},
  224. environment_name => $self->{environment_name} // '',);
  225. scalar $self->{$lang}{deparser}->deparse($ast);
  226. }
  227. #
  228. ## Util functions
  229. #
  230. sub normalize_type {
  231. my ($type) = @_;
  232. if (index($type, 'Sidef::') == 0) {
  233. if ($type =~ /::[0-9]+::/) {
  234. $type = substr($type, $+[0]);
  235. }
  236. else {
  237. $type = substr($type, rindex($type, '::') + 2);
  238. }
  239. }
  240. $type =~ s/^main:://r;
  241. }
  242. sub normalize_method {
  243. my ($type, $method) = ($_[0] =~ /^(.*[^:])::(.*)$/);
  244. normalize_type($type) . '#' . $method;
  245. }
  246. sub jaro {
  247. my ($s, $t) = @_;
  248. my $s_len = length($s);
  249. my $t_len = length($t);
  250. my $match_distance = int(List::Util::max($s_len, $t_len) / 2) - 1;
  251. my @s_matches;
  252. my @t_matches;
  253. my @s = split(//, $s);
  254. my @t = split(//, $t);
  255. my $matches = 0;
  256. foreach my $i (0 .. $s_len - 1) {
  257. my $start = List::Util::max(0, $i - $match_distance);
  258. my $end = List::Util::min($i + $match_distance + 1, $t_len);
  259. foreach my $j ($start .. $end - 1) {
  260. $t_matches[$j] and next;
  261. $s[$i] eq $t[$j] or next;
  262. $s_matches[$i] = 1;
  263. $t_matches[$j] = 1;
  264. $matches++;
  265. last;
  266. }
  267. }
  268. return 0 if $matches == 0;
  269. my $k = 0;
  270. my $trans = 0;
  271. foreach my $i (0 .. $s_len - 1) {
  272. $s_matches[$i] or next;
  273. until ($t_matches[$k]) { ++$k }
  274. $s[$i] eq $t[$k] or ++$trans;
  275. ++$k;
  276. }
  277. #<<<
  278. (($matches / $s_len) + ($matches / $t_len)
  279. + (($matches - $trans / 2) / $matches)) / 3;
  280. #>>>
  281. }
  282. sub jaro_winkler {
  283. my ($s, $t) = @_;
  284. my $distance = jaro($s, $t);
  285. my $prefix = 0;
  286. foreach my $i (0 .. List::Util::min(3, length($s), length($t))) {
  287. substr($s, $i, 1) eq substr($t, $i, 1) ? ++$prefix : last;
  288. }
  289. $distance + $prefix * 0.1 * (1 - $distance);
  290. }
  291. sub best_matches {
  292. my ($name, $set) = @_;
  293. my $max = 0;
  294. my @best;
  295. foreach my $elem (@$set) {
  296. my $dist = sprintf("%.4f", jaro_winkler($elem, $name));
  297. $dist >= 0.8 or next;
  298. if ($dist > $max) {
  299. $max = $dist;
  300. @best = ();
  301. }
  302. push(@best, $elem) if $dist == $max;
  303. }
  304. @best;
  305. }
  306. };
  307. use utf8;
  308. use 5.016;
  309. our $AUTOLOAD;
  310. #
  311. ## UNIVERSAL methods
  312. #
  313. *UNIVERSAL::get_value = sub {
  314. ref($_[0]) eq 'Sidef::Module::OO' || ref($_[0]) eq 'Sidef::Module::Func'
  315. ? $_[0]->{module}
  316. : $_[0];
  317. };
  318. *UNIVERSAL::DESTROY = sub { };
  319. *UNIVERSAL::AUTOLOAD = sub {
  320. my ($self, @args) = @_;
  321. $self = ref($self) if ref($self);
  322. if (index($self, 'Sidef::') == 0 and index($self, 'Sidef::Runtime') != 0) {
  323. eval { require $self =~ s{::}{/}rg . '.pm' };
  324. if ($@) {
  325. if (defined(&main::__load_sidef_module__)) {
  326. main::__load_sidef_module__($self);
  327. }
  328. else {
  329. die "[AUTOLOAD] $@";
  330. }
  331. }
  332. if (defined(&$AUTOLOAD)) {
  333. goto &$AUTOLOAD;
  334. }
  335. }
  336. my @caller = caller(1);
  337. my $from = Sidef::normalize_method($caller[3]);
  338. $from = $from eq '#' ? 'main()' : "$from()";
  339. my $table = do { no strict 'refs'; \%{$self . '::'} };
  340. my @methods = grep { !ref($table->{$_}) and defined(&{$table->{$_}}) } keys(%$table);
  341. my $method = Sidef::normalize_method($AUTOLOAD);
  342. my $name = substr($method, rindex($method, '#') + 1);
  343. my @candidates = Sidef::best_matches($name, \@methods);
  344. die( "[AUTOLOAD] Undefined method `"
  345. . $method . q{'}
  346. . " called from $from\n"
  347. . (@candidates ? ("[?] Did you mean: " . join("\n" . (' ' x 18), sort(@candidates)) . "\n") : ''));
  348. return;
  349. };
  350. 1;