eyeling 1.34.3 → 1.34.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (296) hide show
  1. package/README.md +7 -9
  2. package/docs/eyelang-guide.md +6 -11
  3. package/docs/eyelang-language-reference.md +47 -36
  4. package/examples/eyelang/bayes-therapy.pl +4 -4
  5. package/examples/eyelang/output/reusable-builtins.pl +5 -0
  6. package/examples/eyelang/output/term-tools.pl +6 -0
  7. package/examples/eyelang/reusable-builtins.pl +32 -0
  8. package/examples/eyelang/term-tools.pl +23 -0
  9. package/lib/eyelang/builtins/arithmetic.js +19 -6
  10. package/lib/eyelang/builtins/control.js +12 -0
  11. package/lib/eyelang/builtins/lists.js +146 -7
  12. package/lib/eyelang/builtins/registry.js +2 -3
  13. package/lib/eyelang/builtins/strings.js +165 -1
  14. package/lib/eyelang/builtins/terms.js +66 -0
  15. package/package.json +1 -1
  16. package/test/eyelang/conformance/README.md +9 -11
  17. package/test/eyelang/conformance/cases/001_fact_output.pl +4 -0
  18. package/test/eyelang/conformance/cases/002_rule_recursion.pl +7 -0
  19. package/test/eyelang/conformance/cases/002_rule_recursion.query +1 -0
  20. package/test/eyelang/conformance/cases/003_terms_and_readback.pl +16 -0
  21. package/test/eyelang/conformance/cases/003_terms_and_readback.query +1 -0
  22. package/test/eyelang/conformance/cases/004_conjunction_and_parentheses.pl +5 -0
  23. package/test/eyelang/conformance/cases/004_conjunction_and_parentheses.query +1 -0
  24. package/test/eyelang/conformance/cases/005_list_deconstruction.pl +6 -0
  25. package/test/eyelang/conformance/cases/005_list_deconstruction.query +1 -0
  26. package/test/eyelang/conformance/cases/006_comma_formula_data.pl +4 -0
  27. package/test/eyelang/conformance/cases/006_comma_formula_data.query +1 -0
  28. package/test/eyelang/conformance/cases/007_anonymous_variables.pl +5 -0
  29. package/test/eyelang/conformance/cases/007_anonymous_variables.query +1 -0
  30. package/test/eyelang/conformance/cases/008_graphic_atoms.pl +6 -0
  31. package/test/eyelang/conformance/cases/008_graphic_atoms.query +1 -0
  32. package/test/eyelang/conformance/cases/009_comments_and_whitespace.pl +5 -0
  33. package/test/eyelang/conformance/cases/009_comments_and_whitespace.query +1 -0
  34. package/test/eyelang/conformance/cases/010_variable_scope_and_reuse.pl +8 -0
  35. package/test/eyelang/conformance/cases/010_variable_scope_and_reuse.query +1 -0
  36. package/test/eyelang/conformance/cases/011_predicate_arity.pl +6 -0
  37. package/test/eyelang/conformance/cases/011_predicate_arity.query +1 -0
  38. package/test/eyelang/conformance/cases/012_nested_compound_unification.pl +5 -0
  39. package/test/eyelang/conformance/cases/012_nested_compound_unification.query +1 -0
  40. package/test/eyelang/conformance/cases/013_multiple_clauses_order.pl +6 -0
  41. package/test/eyelang/conformance/cases/013_multiple_clauses_order.query +1 -0
  42. package/test/eyelang/conformance/cases/014_failure_filters_answers.pl +7 -0
  43. package/test/eyelang/conformance/cases/014_failure_filters_answers.query +1 -0
  44. package/test/eyelang/conformance/cases/015_improper_list_unification.pl +6 -0
  45. package/test/eyelang/conformance/cases/015_improper_list_unification.query +1 -0
  46. package/test/eyelang/conformance/cases/016_zero_arity_compound.pl +4 -0
  47. package/test/eyelang/conformance/cases/016_zero_arity_compound.query +1 -0
  48. package/test/eyelang/conformance/cases/017_three_step_recursion.pl +8 -0
  49. package/test/eyelang/conformance/cases/017_three_step_recursion.query +1 -0
  50. package/test/eyelang/conformance/cases/018_quoted_atom_readback.pl +6 -0
  51. package/test/eyelang/conformance/cases/018_quoted_atom_readback.query +1 -0
  52. package/test/eyelang/conformance/cases/019_parenthesized_three_conjuncts.pl +7 -0
  53. package/test/eyelang/conformance/cases/019_parenthesized_three_conjuncts.query +1 -0
  54. package/test/eyelang/conformance/cases/020_nested_list_terms.pl +5 -0
  55. package/test/eyelang/conformance/cases/020_nested_list_terms.query +1 -0
  56. package/test/eyelang/conformance/cases/021_repeated_variable_head.pl +7 -0
  57. package/test/eyelang/conformance/cases/022_rule_head_structure.pl +5 -0
  58. package/test/eyelang/conformance/cases/023_quoted_escapes_readback.pl +5 -0
  59. package/test/eyelang/conformance/cases/024_numeric_literal_readback.pl +6 -0
  60. package/test/eyelang/conformance/cases/025_body_parentheses_with_formula_data.pl +5 -0
  61. package/test/eyelang/conformance/cases/026_underscore_named_variable_reuse.pl +5 -0
  62. package/test/eyelang/conformance/cases/027_default_derived_output.pl +5 -0
  63. package/test/eyelang/conformance/cases/028_materialize_focus.pl +5 -0
  64. package/test/eyelang/conformance/cases/029_arithmetic_and_comparison.pl +11 -0
  65. package/test/eyelang/conformance/cases/029_arithmetic_and_comparison.query +1 -0
  66. package/test/eyelang/conformance/cases/030_strings_and_atoms.pl +4 -0
  67. package/test/eyelang/conformance/cases/030_strings_and_atoms.query +1 -0
  68. package/test/eyelang/conformance/cases/031_lists_aggregation_ordering.pl +10 -0
  69. package/test/eyelang/conformance/cases/031_lists_aggregation_ordering.query +1 -0
  70. package/test/eyelang/conformance/cases/032_holds_parts.pl +4 -0
  71. package/test/eyelang/conformance/cases/033_negation_once_generators.pl +7 -0
  72. package/test/eyelang/conformance/cases/033_negation_once_generators.query +1 -0
  73. package/test/eyelang/conformance/cases/034_equality_and_inequality.pl +6 -0
  74. package/test/eyelang/conformance/cases/034_equality_and_inequality.query +1 -0
  75. package/test/eyelang/conformance/cases/035_list_relations.pl +5 -0
  76. package/test/eyelang/conformance/cases/035_list_relations.query +1 -0
  77. package/test/eyelang/conformance/cases/036_append_splits.pl +3 -0
  78. package/test/eyelang/conformance/cases/036_append_splits.query +1 -0
  79. package/test/eyelang/conformance/cases/037_matching_and_comparison.pl +7 -0
  80. package/test/eyelang/conformance/cases/037_matching_and_comparison.query +1 -0
  81. package/test/eyelang/conformance/cases/038_memoize_declaration.pl +8 -0
  82. package/test/eyelang/conformance/cases/038_memoize_declaration.query +1 -0
  83. package/test/eyelang/conformance/cases/039_numeric_functions.pl +9 -0
  84. package/test/eyelang/conformance/cases/039_numeric_functions.query +1 -0
  85. package/test/eyelang/conformance/cases/040_between_enumeration.pl +3 -0
  86. package/test/eyelang/conformance/cases/040_between_enumeration.query +1 -0
  87. package/test/eyelang/conformance/cases/041_smallest_divisor.pl +3 -0
  88. package/test/eyelang/conformance/cases/041_smallest_divisor.query +1 -0
  89. package/test/eyelang/conformance/cases/042_negation_filter.pl +7 -0
  90. package/test/eyelang/conformance/cases/042_negation_filter.query +1 -0
  91. package/test/eyelang/conformance/cases/043_once_user_predicate.pl +5 -0
  92. package/test/eyelang/conformance/cases/043_once_user_predicate.query +1 -0
  93. package/test/eyelang/conformance/cases/044_findall_user_goal.pl +6 -0
  94. package/test/eyelang/conformance/cases/044_findall_user_goal.query +1 -0
  95. package/test/eyelang/conformance/cases/045_sort_deduplicates_atoms.pl +3 -0
  96. package/test/eyelang/conformance/cases/045_sort_deduplicates_atoms.query +1 -0
  97. package/test/eyelang/conformance/cases/046_append_bound_prefix_suffix.pl +4 -0
  98. package/test/eyelang/conformance/cases/046_append_bound_prefix_suffix.query +1 -0
  99. package/test/eyelang/conformance/cases/047_nth0_index_generation.pl +3 -0
  100. package/test/eyelang/conformance/cases/047_nth0_index_generation.query +1 -0
  101. package/test/eyelang/conformance/cases/048_set_nth0_edges.pl +4 -0
  102. package/test/eyelang/conformance/cases/048_set_nth0_edges.query +1 -0
  103. package/test/eyelang/conformance/cases/049_select_duplicate_occurrences.pl +3 -0
  104. package/test/eyelang/conformance/cases/049_select_duplicate_occurrences.query +1 -0
  105. package/test/eyelang/conformance/cases/050_not_member_filter.pl +6 -0
  106. package/test/eyelang/conformance/cases/050_not_member_filter.query +1 -0
  107. package/test/eyelang/conformance/cases/051_nested_holds_parts.pl +4 -0
  108. package/test/eyelang/conformance/cases/052_holds_member.pl +3 -0
  109. package/test/eyelang/conformance/cases/053_materialize_excludes_source_fact.pl +6 -0
  110. package/test/eyelang/conformance/cases/054_numeric_and_lexical_comparison.pl +5 -0
  111. package/test/eyelang/conformance/cases/054_numeric_and_lexical_comparison.query +1 -0
  112. package/test/eyelang/conformance/cases/055_string_matching_filters.pl +6 -0
  113. package/test/eyelang/conformance/cases/055_string_matching_filters.query +1 -0
  114. package/test/eyelang/conformance/cases/056_string_and_atom_concat.pl +3 -0
  115. package/test/eyelang/conformance/cases/056_string_and_atom_concat.query +1 -0
  116. package/test/eyelang/conformance/cases/057_countall_empty_and_nonempty.pl +4 -0
  117. package/test/eyelang/conformance/cases/057_countall_empty_and_nonempty.query +1 -0
  118. package/test/eyelang/conformance/cases/058_sumall_numeric_template.pl +5 -0
  119. package/test/eyelang/conformance/cases/058_sumall_numeric_template.query +1 -0
  120. package/test/eyelang/conformance/cases/059_aggregate_min_template.pl +5 -0
  121. package/test/eyelang/conformance/cases/059_aggregate_min_template.query +1 -0
  122. package/test/eyelang/conformance/cases/060_aggregate_max_compound_key.pl +5 -0
  123. package/test/eyelang/conformance/cases/060_aggregate_max_compound_key.query +1 -0
  124. package/test/eyelang/conformance/cases/061_date_difference.pl +4 -0
  125. package/test/eyelang/conformance/cases/062_reusable_numeric_builtins.pl +10 -0
  126. package/test/eyelang/conformance/cases/063_reusable_list_builtins.pl +11 -0
  127. package/test/eyelang/conformance/cases/064_reusable_string_builtins.pl +12 -0
  128. package/test/eyelang/conformance/cases/065_reusable_term_control_builtins.pl +11 -0
  129. package/test/eyelang/conformance/cases/066_numeric_edges.pl +14 -0
  130. package/test/eyelang/conformance/cases/067_list_edges.pl +10 -0
  131. package/test/eyelang/conformance/cases/068_list_generation_order.pl +7 -0
  132. package/test/eyelang/conformance/cases/069_list_summaries_and_sets.pl +9 -0
  133. package/test/eyelang/conformance/cases/070_matches_named_captures.pl +13 -0
  134. package/test/eyelang/conformance/cases/071_string_edges.pl +10 -0
  135. package/test/eyelang/conformance/cases/072_string_conversions.pl +10 -0
  136. package/test/eyelang/conformance/cases/073_term_introspection_edges.pl +10 -0
  137. package/test/eyelang/conformance/cases/074_forall_edges.pl +10 -0
  138. package/test/eyelang/conformance/cases/075_aggregation_edges.pl +12 -0
  139. package/test/eyelang/conformance/cases/076_composed_reusable_builtins.pl +8 -0
  140. package/test/eyelang/conformance/cases/077_recursive_path_with_lists.pl +10 -0
  141. package/test/eyelang/conformance/cases/078_mutual_recursion_with_arithmetic.pl +7 -0
  142. package/test/eyelang/conformance/cases/079_big_integer_arithmetic.pl +8 -0
  143. package/test/eyelang/conformance/cases/080_rounding_modes.pl +10 -0
  144. package/test/eyelang/conformance/cases/081_zero_safe_numeric_functions.pl +9 -0
  145. package/test/eyelang/conformance/cases/082_comparison_semantics.pl +10 -0
  146. package/test/eyelang/conformance/cases/083_between_modes_and_empty_ranges.pl +8 -0
  147. package/test/eyelang/conformance/cases/084_append_and_select_composition.pl +7 -0
  148. package/test/eyelang/conformance/cases/085_nth_and_update_edges.pl +8 -0
  149. package/test/eyelang/conformance/cases/086_slicing_pipeline.pl +10 -0
  150. package/test/eyelang/conformance/cases/087_sort_reverse_length.pl +8 -0
  151. package/test/eyelang/conformance/cases/088_list_summaries_failures.pl +8 -0
  152. package/test/eyelang/conformance/cases/089_string_split_join_pipeline.pl +7 -0
  153. package/test/eyelang/conformance/cases/090_string_substring_replace_edges.pl +9 -0
  154. package/test/eyelang/conformance/cases/091_string_case_and_trim.pl +7 -0
  155. package/test/eyelang/conformance/cases/092_scalar_string_conversions.pl +9 -0
  156. package/test/eyelang/conformance/cases/093_regex_named_captures_context.pl +8 -0
  157. package/test/eyelang/conformance/cases/094_context_holds_enumeration.pl +7 -0
  158. package/test/eyelang/conformance/cases/095_term_introspection_roundtrip.pl +8 -0
  159. package/test/eyelang/conformance/cases/096_functor_scalar_edges.pl +8 -0
  160. package/test/eyelang/conformance/cases/097_control_negation_once_forall.pl +13 -0
  161. package/test/eyelang/conformance/cases/098_aggregation_nested_templates.pl +11 -0
  162. package/test/eyelang/conformance/cases/099_materialize_multiple_arities.pl +8 -0
  163. package/test/eyelang/conformance/cases/100_reusable_builtin_workflow.pl +10 -0
  164. package/test/eyelang/conformance/cases/extension/036_reusable_numeric_builtins.pl +10 -0
  165. package/test/eyelang/conformance/cases/extension/037_reusable_list_builtins.pl +11 -0
  166. package/test/eyelang/conformance/cases/extension/038_reusable_string_builtins.pl +12 -0
  167. package/test/eyelang/conformance/cases/extension/039_reusable_term_control_builtins.pl +11 -0
  168. package/test/eyelang/conformance/expected/001_fact_output.pl +1 -0
  169. package/test/eyelang/conformance/expected/002_rule_recursion.pl +2 -0
  170. package/test/eyelang/conformance/expected/003_terms_and_readback.pl +13 -0
  171. package/test/eyelang/conformance/expected/004_conjunction_and_parentheses.pl +1 -0
  172. package/test/eyelang/conformance/expected/005_list_deconstruction.pl +2 -0
  173. package/test/eyelang/conformance/expected/006_comma_formula_data.pl +1 -0
  174. package/test/eyelang/conformance/expected/007_anonymous_variables.pl +1 -0
  175. package/test/eyelang/conformance/expected/008_graphic_atoms.pl +3 -0
  176. package/test/eyelang/conformance/expected/009_comments_and_whitespace.pl +2 -0
  177. package/test/eyelang/conformance/expected/010_variable_scope_and_reuse.pl +2 -0
  178. package/test/eyelang/conformance/expected/011_predicate_arity.pl +2 -0
  179. package/test/eyelang/conformance/expected/012_nested_compound_unification.pl +2 -0
  180. package/test/eyelang/conformance/expected/013_multiple_clauses_order.pl +2 -0
  181. package/test/eyelang/conformance/expected/014_failure_filters_answers.pl +1 -0
  182. package/test/eyelang/conformance/expected/015_improper_list_unification.pl +3 -0
  183. package/test/eyelang/conformance/expected/016_zero_arity_compound.pl +1 -0
  184. package/test/eyelang/conformance/expected/017_three_step_recursion.pl +3 -0
  185. package/test/eyelang/conformance/expected/018_quoted_atom_readback.pl +3 -0
  186. package/test/eyelang/conformance/expected/019_parenthesized_three_conjuncts.pl +1 -0
  187. package/test/eyelang/conformance/expected/020_nested_list_terms.pl +2 -0
  188. package/test/eyelang/conformance/expected/021_repeated_variable_head.pl +2 -0
  189. package/test/eyelang/conformance/expected/022_rule_head_structure.pl +2 -0
  190. package/test/eyelang/conformance/expected/023_quoted_escapes_readback.pl +2 -0
  191. package/test/eyelang/conformance/expected/024_numeric_literal_readback.pl +3 -0
  192. package/test/eyelang/conformance/expected/025_body_parentheses_with_formula_data.pl +1 -0
  193. package/test/eyelang/conformance/expected/026_underscore_named_variable_reuse.pl +1 -0
  194. package/test/eyelang/conformance/expected/027_default_derived_output.pl +3 -0
  195. package/test/eyelang/conformance/expected/028_materialize_focus.pl +1 -0
  196. package/test/eyelang/conformance/expected/029_arithmetic_and_comparison.pl +9 -0
  197. package/test/eyelang/conformance/expected/030_strings_and_atoms.pl +2 -0
  198. package/test/eyelang/conformance/expected/031_lists_aggregation_ordering.pl +9 -0
  199. package/test/eyelang/conformance/expected/032_holds_parts.pl +3 -0
  200. package/test/eyelang/conformance/expected/033_negation_once_generators.pl +2 -0
  201. package/test/eyelang/conformance/expected/034_equality_and_inequality.pl +4 -0
  202. package/test/eyelang/conformance/expected/035_list_relations.pl +4 -0
  203. package/test/eyelang/conformance/expected/036_append_splits.pl +3 -0
  204. package/test/eyelang/conformance/expected/037_matching_and_comparison.pl +5 -0
  205. package/test/eyelang/conformance/expected/038_memoize_declaration.pl +2 -0
  206. package/test/eyelang/conformance/expected/039_numeric_functions.pl +7 -0
  207. package/test/eyelang/conformance/expected/040_between_enumeration.pl +3 -0
  208. package/test/eyelang/conformance/expected/041_smallest_divisor.pl +1 -0
  209. package/test/eyelang/conformance/expected/042_negation_filter.pl +2 -0
  210. package/test/eyelang/conformance/expected/043_once_user_predicate.pl +1 -0
  211. package/test/eyelang/conformance/expected/044_findall_user_goal.pl +1 -0
  212. package/test/eyelang/conformance/expected/045_sort_deduplicates_atoms.pl +1 -0
  213. package/test/eyelang/conformance/expected/046_append_bound_prefix_suffix.pl +2 -0
  214. package/test/eyelang/conformance/expected/047_nth0_index_generation.pl +1 -0
  215. package/test/eyelang/conformance/expected/048_set_nth0_edges.pl +2 -0
  216. package/test/eyelang/conformance/expected/049_select_duplicate_occurrences.pl +2 -0
  217. package/test/eyelang/conformance/expected/050_not_member_filter.pl +1 -0
  218. package/test/eyelang/conformance/expected/051_nested_holds_parts.pl +3 -0
  219. package/test/eyelang/conformance/expected/052_holds_member.pl +2 -0
  220. package/test/eyelang/conformance/expected/053_materialize_excludes_source_fact.pl +1 -0
  221. package/test/eyelang/conformance/expected/054_numeric_and_lexical_comparison.pl +3 -0
  222. package/test/eyelang/conformance/expected/055_string_matching_filters.pl +2 -0
  223. package/test/eyelang/conformance/expected/056_string_and_atom_concat.pl +1 -0
  224. package/test/eyelang/conformance/expected/057_countall_empty_and_nonempty.pl +1 -0
  225. package/test/eyelang/conformance/expected/058_sumall_numeric_template.pl +1 -0
  226. package/test/eyelang/conformance/expected/059_aggregate_min_template.pl +1 -0
  227. package/test/eyelang/conformance/expected/060_aggregate_max_compound_key.pl +1 -0
  228. package/test/eyelang/conformance/expected/061_date_difference.pl +2 -0
  229. package/test/eyelang/conformance/expected/062_reusable_numeric_builtins.pl +8 -0
  230. package/test/eyelang/conformance/expected/063_reusable_list_builtins.pl +9 -0
  231. package/test/eyelang/conformance/expected/064_reusable_string_builtins.pl +10 -0
  232. package/test/eyelang/conformance/expected/065_reusable_term_control_builtins.pl +6 -0
  233. package/test/eyelang/conformance/expected/066_numeric_edges.pl +12 -0
  234. package/test/eyelang/conformance/expected/067_list_edges.pl +8 -0
  235. package/test/eyelang/conformance/expected/068_list_generation_order.pl +10 -0
  236. package/test/eyelang/conformance/expected/069_list_summaries_and_sets.pl +7 -0
  237. package/test/eyelang/conformance/expected/070_matches_named_captures.pl +1 -0
  238. package/test/eyelang/conformance/expected/071_string_edges.pl +8 -0
  239. package/test/eyelang/conformance/expected/072_string_conversions.pl +8 -0
  240. package/test/eyelang/conformance/expected/073_term_introspection_edges.pl +8 -0
  241. package/test/eyelang/conformance/expected/074_forall_edges.pl +5 -0
  242. package/test/eyelang/conformance/expected/075_aggregation_edges.pl +7 -0
  243. package/test/eyelang/conformance/expected/076_composed_reusable_builtins.pl +5 -0
  244. package/test/eyelang/conformance/expected/077_recursive_path_with_lists.pl +3 -0
  245. package/test/eyelang/conformance/expected/078_mutual_recursion_with_arithmetic.pl +7 -0
  246. package/test/eyelang/conformance/expected/079_big_integer_arithmetic.pl +6 -0
  247. package/test/eyelang/conformance/expected/080_rounding_modes.pl +8 -0
  248. package/test/eyelang/conformance/expected/081_zero_safe_numeric_functions.pl +7 -0
  249. package/test/eyelang/conformance/expected/082_comparison_semantics.pl +8 -0
  250. package/test/eyelang/conformance/expected/083_between_modes_and_empty_ranges.pl +8 -0
  251. package/test/eyelang/conformance/expected/084_append_and_select_composition.pl +10 -0
  252. package/test/eyelang/conformance/expected/085_nth_and_update_edges.pl +8 -0
  253. package/test/eyelang/conformance/expected/086_slicing_pipeline.pl +7 -0
  254. package/test/eyelang/conformance/expected/087_sort_reverse_length.pl +6 -0
  255. package/test/eyelang/conformance/expected/088_list_summaries_failures.pl +6 -0
  256. package/test/eyelang/conformance/expected/089_string_split_join_pipeline.pl +4 -0
  257. package/test/eyelang/conformance/expected/090_string_substring_replace_edges.pl +7 -0
  258. package/test/eyelang/conformance/expected/091_string_case_and_trim.pl +5 -0
  259. package/test/eyelang/conformance/expected/092_scalar_string_conversions.pl +7 -0
  260. package/test/eyelang/conformance/expected/093_regex_named_captures_context.pl +5 -0
  261. package/test/eyelang/conformance/expected/094_context_holds_enumeration.pl +8 -0
  262. package/test/eyelang/conformance/expected/095_term_introspection_roundtrip.pl +6 -0
  263. package/test/eyelang/conformance/expected/096_functor_scalar_edges.pl +6 -0
  264. package/test/eyelang/conformance/expected/097_control_negation_once_forall.pl +5 -0
  265. package/test/eyelang/conformance/expected/098_aggregation_nested_templates.pl +5 -0
  266. package/test/eyelang/conformance/expected/099_materialize_multiple_arities.pl +4 -0
  267. package/test/eyelang/conformance/expected/100_reusable_builtin_workflow.pl +6 -0
  268. package/test/eyelang/conformance/expected/extension/036_reusable_numeric_builtins.out +8 -0
  269. package/test/eyelang/conformance/expected/extension/037_reusable_list_builtins.out +9 -0
  270. package/test/eyelang/conformance/expected/extension/038_reusable_string_builtins.out +10 -0
  271. package/test/eyelang/conformance/expected/extension/039_reusable_term_control_builtins.out +6 -0
  272. package/test/eyelang/run-conformance.mjs +20 -22
  273. package/examples/eyelang/collatz-1000.pl +0 -14
  274. package/examples/eyelang/complex-matrix-stability.pl +0 -45
  275. package/examples/eyelang/gcd-bezout-identity.pl +0 -48
  276. package/examples/eyelang/goldbach-1000.pl +0 -185
  277. package/examples/eyelang/kaprekar.pl +0 -32
  278. package/examples/eyelang/matrix.pl +0 -296
  279. package/examples/eyelang/output/collatz-1000.pl +0 -1000
  280. package/examples/eyelang/output/complex-matrix-stability.pl +0 -5
  281. package/examples/eyelang/output/gcd-bezout-identity.pl +0 -36
  282. package/examples/eyelang/output/goldbach-1000.pl +0 -667
  283. package/examples/eyelang/output/kaprekar.pl +0 -8
  284. package/examples/eyelang/output/matrix.pl +0 -10
  285. package/lib/eyelang/builtins/matrix.js +0 -226
  286. package/lib/eyelang/builtins/number-theory.js +0 -114
  287. package/test/eyelang/conformance/cases/extension/036_extended_gcd.pl +0 -3
  288. package/test/eyelang/conformance/cases/extension/037_collatz_trajectory.pl +0 -3
  289. package/test/eyelang/conformance/cases/extension/038_kaprekar_steps.pl +0 -3
  290. package/test/eyelang/conformance/cases/extension/039_goldbach_pair.pl +0 -3
  291. package/test/eyelang/conformance/cases/extension/040_matrix_operations.pl +0 -5
  292. package/test/eyelang/conformance/expected/extension/036_extended_gcd.out +0 -1
  293. package/test/eyelang/conformance/expected/extension/037_collatz_trajectory.out +0 -1
  294. package/test/eyelang/conformance/expected/extension/038_kaprekar_steps.out +0 -1
  295. package/test/eyelang/conformance/expected/extension/039_goldbach_pair.out +0 -2
  296. package/test/eyelang/conformance/expected/extension/040_matrix_operations.out +0 -3
@@ -1,226 +0,0 @@
1
- // Small dense-matrix arithmetic builtins.
2
- // These are reusable numeric kernels for ground matrices; when arguments are not
3
- // ground proper numeric lists, user-defined matrix predicates remain available.
4
- import { deref, isDecimalInteger, listFromItems, numberTerm, numberTextFromDouble, parseFiniteNumber, properListItems, unify } from '../term.js';
5
-
6
- export const matrixBuiltins = {
7
- register(registry) {
8
- registry.add('matrix_sum', 2, matrixSum, { deterministic: true, fallbackWhenNotReady: true, ready: matrixPairReady });
9
- registry.add('matrix_multiply', 2, matrixMultiply, { deterministic: true, fallbackWhenNotReady: true, ready: matrixPairReady });
10
- registry.add('cholesky_decomposition', 2, choleskyDecomposition, { deterministic: true, fallbackWhenNotReady: true, ready: firstMatrixReady });
11
- registry.add('determinant', 2, determinant, { deterministic: true, fallbackWhenNotReady: true, ready: firstMatrixReady });
12
- registry.add('matrix_inv_triang', 2, matrixInvTriang, { deterministic: true, fallbackWhenNotReady: true, ready: firstMatrixReady });
13
- registry.add('matrix_inversion', 2, matrixInversion, { deterministic: true, fallbackWhenNotReady: true, ready: firstMatrixReady });
14
- }
15
- };
16
-
17
- function firstMatrixReady(goal, env) { return parseMatrix(goal.args[0], env) !== null; }
18
- function matrixPairReady(goal, env) { return parseMatrixPair(goal.args[0], env) !== null; }
19
-
20
- function* matrixSum({ goal, env }) {
21
- const pair = parseMatrixPair(goal.args[0], env);
22
- if (!pair) return;
23
- const [a, b] = pair;
24
- if (!sameShape(a, b)) return;
25
- const out = a.map((row, i) => row.map((cell, j) => addNum(cell, b[i][j])));
26
- yield* unifyMatrix(goal.args[1], out, env);
27
- }
28
-
29
- function* matrixMultiply({ goal, env }) {
30
- const pair = parseMatrixPair(goal.args[0], env);
31
- if (!pair) return;
32
- const out = multiplyMatrices(pair[0], pair[1]);
33
- if (!out) return;
34
- yield* unifyMatrix(goal.args[1], out, env);
35
- }
36
-
37
- function* choleskyDecomposition({ goal, env }) {
38
- const matrix = parseMatrix(goal.args[0], env);
39
- const out = cholesky(matrix);
40
- if (!out) return;
41
- yield* unifyMatrix(goal.args[1], out, env);
42
- }
43
-
44
- function* determinant({ goal, env }) {
45
- const matrix = parseMatrix(goal.args[0], env);
46
- const l = cholesky(matrix);
47
- if (!l) return;
48
- let prod = num(1n);
49
- for (let i = l.length - 1; i >= 0; i--) prod = mulNum(l[i][i], prod);
50
- const det = mulNum(prod, prod);
51
- const next = env.clone();
52
- if (unify(goal.args[1], numTerm(det), next)) yield next;
53
- }
54
-
55
- function* matrixInvTriang({ goal, env }) {
56
- const matrix = parseMatrix(goal.args[0], env);
57
- const out = invertLowerTriangular(matrix);
58
- if (!out) return;
59
- yield* unifyMatrix(goal.args[1], out, env);
60
- }
61
-
62
- function* matrixInversion({ goal, env }) {
63
- const matrix = parseMatrix(goal.args[0], env);
64
- const l = cholesky(matrix);
65
- if (!l) return;
66
- const li = invertLowerTriangular(l);
67
- if (!li) return;
68
- const lit = transpose(li);
69
- const out = multiplyMatrices(lit, li);
70
- if (!out) return;
71
- yield* unifyMatrix(goal.args[1], out, env);
72
- }
73
-
74
- function cholesky(a) {
75
- if (!isSquare(a)) return null;
76
- const n = a.length;
77
- const l = Array.from({ length: n }, () => Array.from({ length: n }, () => num(0n)));
78
- for (let i = 0; i < n; i++) {
79
- for (let j = 0; j <= i; j++) {
80
- let sum = num(0n);
81
- for (let k = 0; k < j; k++) sum = addNum(sum, mulNum(l[i][k], l[j][k]));
82
- if (j === i) {
83
- const v2 = subNum(a[i][i], sum);
84
- const v = Math.sqrt(toNumber(v2));
85
- if (!Number.isFinite(v)) return null;
86
- l[i][j] = num(v);
87
- } else {
88
- const numerator = subNum(a[i][j], sum);
89
- l[i][j] = divNum(numerator, l[j][j]);
90
- }
91
- }
92
- }
93
- return l;
94
- }
95
-
96
- function invertLowerTriangular(l) {
97
- if (!isSquare(l)) return null;
98
- const n = l.length;
99
- const previous = [];
100
- const out = [];
101
- for (let i = 0; i < n; i++) {
102
- const row = [];
103
- for (let j = 0; j < n; j++) {
104
- if (j > i) row.push(num(0n));
105
- else if (i === j) row.push(divNum(num(1), l[i][i]));
106
- else {
107
- let sum = num(0n);
108
- for (let k = j; k < i; k++) sum = addNum(sum, mulNum(l[i][k], previous[k][j]));
109
- row.push(divNum(negNum(sum), l[i][i]));
110
- }
111
- }
112
- previous.push(row);
113
- out.push(row);
114
- }
115
- return out;
116
- }
117
-
118
- function multiplyMatrices(a, b) {
119
- if (!a.length || !b.length || !a[0].length || !b[0].length) return null;
120
- const rows = a.length, inner = a[0].length, cols = b[0].length;
121
- if (b.length !== inner || a.some((row) => row.length !== inner) || b.some((row) => row.length !== cols)) return null;
122
- const out = [];
123
- for (let i = 0; i < rows; i++) {
124
- const row = [];
125
- for (let j = 0; j < cols; j++) {
126
- let total = num(0n);
127
- for (let k = inner - 1; k >= 0; k--) total = addNum(mulNum(a[i][k], b[k][j]), total);
128
- row.push(total);
129
- }
130
- out.push(row);
131
- }
132
- return out;
133
- }
134
-
135
- function transpose(a) {
136
- if (!a.length) return [];
137
- return a[0].map((_, i) => a.map((row) => row[i]));
138
- }
139
-
140
- function sameShape(a, b) {
141
- return a.length === b.length && a.every((row, i) => row.length === b[i].length);
142
- }
143
-
144
- function isSquare(a) {
145
- return Array.isArray(a) && a.length > 0 && a.every((row) => row.length === a.length);
146
- }
147
-
148
- function parseMatrixPair(term, env) {
149
- const pair = properListItems(term, env);
150
- if (!pair || pair.length !== 2) return null;
151
- const a = parseMatrix(pair[0], env), b = parseMatrix(pair[1], env);
152
- return a && b ? [a, b] : null;
153
- }
154
-
155
- function parseMatrix(term, env) {
156
- const rows = properListItems(term, env);
157
- if (!rows) return null;
158
- const matrix = [];
159
- let width = null;
160
- for (const rowTerm of rows) {
161
- const rowItems = properListItems(rowTerm, env);
162
- if (!rowItems) return null;
163
- if (width == null) width = rowItems.length;
164
- else if (width !== rowItems.length) return null;
165
- const row = [];
166
- for (const item of rowItems) {
167
- const n = parseNum(item, env);
168
- if (!n) return null;
169
- row.push(n);
170
- }
171
- matrix.push(row);
172
- }
173
- return matrix;
174
- }
175
-
176
- function parseNum(term, env) {
177
- const value = deref(term, env);
178
- if (value.type !== 'number') return null;
179
- if (isDecimalInteger(value.name)) return num(BigInt(value.name));
180
- const f = parseFiniteNumber(value.name);
181
- return f == null ? null : num(f);
182
- }
183
-
184
- function* unifyMatrix(target, matrix, env) {
185
- const next = env.clone();
186
- if (unify(target, matrixTerm(matrix), next)) yield next;
187
- }
188
-
189
- function matrixTerm(matrix) {
190
- return listFromItems(matrix.map((row) => listFromItems(row.map(numTerm))));
191
- }
192
-
193
- function numTerm(value) {
194
- if (value.kind === 'int') return numberTerm(value.value.toString());
195
- return numberTerm(numberTextFromDouble(value.value));
196
- }
197
-
198
- function num(value) {
199
- return typeof value === 'bigint' ? { kind: 'int', value } : { kind: 'float', value: Number(value) };
200
- }
201
-
202
- function toNumber(value) {
203
- return value.kind === 'int' ? Number(value.value) : value.value;
204
- }
205
-
206
- function addNum(a, b) {
207
- if (a.kind === 'int' && b.kind === 'int') return num(a.value + b.value);
208
- return num(toNumber(a) + toNumber(b));
209
- }
210
- function subNum(a, b) {
211
- if (a.kind === 'int' && b.kind === 'int') return num(a.value - b.value);
212
- return num(toNumber(a) - toNumber(b));
213
- }
214
- function mulNum(a, b) {
215
- if (a.kind === 'int' && b.kind === 'int') return num(a.value * b.value);
216
- return num(toNumber(a) * toNumber(b));
217
- }
218
- function divNum(a, b) {
219
- if (b.kind === 'int' && b.value === 0n) return num(Number.NaN);
220
- if (b.kind === 'float' && b.value === 0) return num(Number.NaN);
221
- if (a.kind === 'int' && b.kind === 'int' && a.value % b.value === 0n) return num(a.value / b.value);
222
- return num(toNumber(a) / toNumber(b));
223
- }
224
- function negNum(a) {
225
- return a.kind === 'int' ? num(-a.value) : num(-a.value);
226
- }
@@ -1,114 +0,0 @@
1
- // Integer number-theory helpers for examples that would otherwise encode
2
- // arithmetic algorithms with many small recursive rule calls.
3
- // The predicates are intentionally general: extended GCD, Collatz trajectories,
4
- // Kaprekar iteration counts, and Goldbach pair generation.
5
- import { deref, listFromItems, numberTerm, unify } from '../term.js';
6
-
7
- export const numberTheoryBuiltins = {
8
- register(registry) {
9
- registry.add('extended_gcd', 5, extendedGcd, { deterministic: true, fallbackWhenNotReady: true, ready: firstTwoIntsReady });
10
- registry.add('collatz_trajectory', 2, collatzTrajectory, { deterministic: true, fallbackWhenNotReady: true, ready: firstIntReady });
11
- registry.add('kaprekar_steps', 2, kaprekarSteps, { deterministic: true, fallbackWhenNotReady: true, ready: firstIntReady });
12
- registry.add('goldbach_pair', 3, goldbachPair, { fallbackWhenNotReady: true, ready: firstIntReady });
13
- }
14
- };
15
-
16
- function firstIntReady(goal, env) { return intValue(goal.args[0], env) !== null; }
17
- function firstTwoIntsReady(goal, env) { return intValue(goal.args[0], env) !== null && intValue(goal.args[1], env) !== null; }
18
-
19
- function* extendedGcd({ goal, env }) {
20
- const a = intValue(goal.args[0], env);
21
- const b = intValue(goal.args[1], env);
22
- if (a === null || b === null) return;
23
- const { gcd, s, t } = egcdSigned(a, b);
24
- const next = env.clone();
25
- if (unify(goal.args[2], numberTerm(gcd.toString()), next) &&
26
- unify(goal.args[3], numberTerm(s.toString()), next) &&
27
- unify(goal.args[4], numberTerm(t.toString()), next)) yield next;
28
- }
29
-
30
- function egcdSigned(a, b) {
31
- const signA = a < 0n ? -1n : 1n;
32
- const signB = b < 0n ? -1n : 1n;
33
- let oldR = absBigInt(a), r = absBigInt(b);
34
- let oldS = 1n, s = 0n;
35
- let oldT = 0n, t = 1n;
36
- while (r !== 0n) {
37
- const q = oldR / r;
38
- [oldR, r] = [r, oldR - q * r];
39
- [oldS, s] = [s, oldS - q * s];
40
- [oldT, t] = [t, oldT - q * t];
41
- }
42
- return { gcd: oldR, s: oldS * signA, t: oldT * signB };
43
- }
44
-
45
- function absBigInt(n) { return n < 0n ? -n : n; }
46
-
47
- function* collatzTrajectory({ goal, env }) {
48
- const n = intValue(goal.args[0], env);
49
- if (n === null || n < 1n) return;
50
- const seen = [];
51
- let current = n;
52
- while (current >= 1n) {
53
- seen.push(numberTerm(current.toString()));
54
- if (current === 1n) break;
55
- current = current % 2n === 0n ? current / 2n : current * 3n + 1n;
56
- if (seen.length > 100000) return;
57
- }
58
- const next = env.clone();
59
- if (unify(goal.args[1], listFromItems(seen), next)) yield next;
60
- }
61
-
62
- function* kaprekarSteps({ goal, env }) {
63
- const n = intValue(goal.args[0], env);
64
- if (n === null || n < 0n || n > 9999n) return;
65
- const count = kaprekarCount(Number(n));
66
- if (count === null) return;
67
- const next = env.clone();
68
- if (unify(goal.args[1], numberTerm(String(count)), next)) yield next;
69
- }
70
-
71
- function kaprekarCount(start) {
72
- if (start === 6174) return 0;
73
- let current = start;
74
- for (let count = 1; count <= 100; count++) {
75
- const digits = String(current).padStart(4, '0').slice(-4).split('').map(Number).sort((a, b) => a - b);
76
- const low = digitsToNumber(digits);
77
- const high = digitsToNumber([...digits].reverse());
78
- current = high - low;
79
- if (current === 6174) return count;
80
- if (current <= 0) return null;
81
- }
82
- return null;
83
- }
84
-
85
- function digitsToNumber(digits) {
86
- return (((digits[0] * 10 + digits[1]) * 10 + digits[2]) * 10 + digits[3]);
87
- }
88
-
89
- function* goldbachPair({ goal, env }) {
90
- const n = intValue(goal.args[0], env);
91
- if (n === null || n < 4n || n % 2n !== 0n || n > BigInt(Number.MAX_SAFE_INTEGER)) return;
92
- const limit = Number(n / 2n);
93
- const total = Number(n);
94
- for (let p = 2; p <= limit; p = p === 2 ? 3 : p + 2) {
95
- const q = total - p;
96
- if (!isPrimeNumber(p) || !isPrimeNumber(q)) continue;
97
- const next = env.clone();
98
- if (unify(goal.args[1], numberTerm(String(p)), next) && unify(goal.args[2], numberTerm(String(q)), next)) yield next;
99
- }
100
- }
101
-
102
- function isPrimeNumber(n) {
103
- if (n < 2) return false;
104
- if (n === 2) return true;
105
- if (n % 2 === 0) return false;
106
- for (let d = 3; d <= Math.floor(Math.sqrt(n)); d += 2) if (n % d === 0) return false;
107
- return true;
108
- }
109
-
110
- function intValue(term, env) {
111
- const value = deref(term, env);
112
- if (value.type !== 'number' || !/^-?\d+$/.test(value.name)) return null;
113
- return BigInt(value.name);
114
- }
@@ -1,3 +0,0 @@
1
- % Reference 9.2: extended_gcd/5 returns gcd and Bezout coefficients.
2
- answer(gcd, result(G, S, T)) :- extended_gcd(240, 46, G, S, T).
3
- materialize(answer, 2).
@@ -1,3 +0,0 @@
1
- % Reference 9.2: collatz_trajectory/2 returns the full path to 1.
2
- answer(path, T) :- collatz_trajectory(6, T).
3
- materialize(answer, 2).
@@ -1,3 +0,0 @@
1
- % Reference 9.2: kaprekar_steps/2 counts iterations to 6174.
2
- answer(steps, N) :- kaprekar_steps(3524, N).
3
- materialize(answer, 2).
@@ -1,3 +0,0 @@
1
- % Reference 9.2: goldbach_pair/3 enumerates prime decompositions in ascending first prime order.
2
- answer(pair, pair(P, Q)) :- goldbach_pair(28, P, Q).
3
- materialize(answer, 2).
@@ -1,5 +0,0 @@
1
- % Reference 9.2: matrix helpers handle small ground numeric matrices.
2
- answer(sum, M) :- matrix_sum([[[1, 2], [3, 4]], [[5, 6], [7, 8]]], M).
3
- answer(product, M) :- matrix_multiply([[[1, 2], [3, 4]], [[2, 0], [1, 2]]], M).
4
- answer(determinant, D) :- determinant([[4, 2], [2, 3]], D).
5
- materialize(answer, 2).
@@ -1 +0,0 @@
1
- answer(gcd, result(2, -9, 47)).
@@ -1 +0,0 @@
1
- answer(path, [6, 3, 10, 5, 16, 8, 4, 2, 1]).
@@ -1,2 +0,0 @@
1
- answer(pair, pair(5, 23)).
2
- answer(pair, pair(11, 17)).
@@ -1,3 +0,0 @@
1
- answer(sum, [[6, 8], [10, 12]]).
2
- answer(product, [[4, 4], [10, 8]]).
3
- answer(determinant, 8.0000000000000018).