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
@@ -0,0 +1,5 @@
1
+ % Reference 5.3, 7.1: unification follows nested compound term structure.
2
+ fact(pair(a, nested(b, [c, d]))).
3
+ answer(middle, X) :- fact(pair(a, nested(X, [c, d]))).
4
+ answer(list_tail, T) :- fact(pair(a, nested(b, [c | T]))).
5
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 6, 7: multiple clauses for one predicate are explored as alternatives.
2
+ color(red).
3
+ color(blue).
4
+ paint(X) :- color(X).
5
+ answer(color, X) :- paint(X).
6
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 7.2: a failing subgoal removes that candidate answer.
2
+ candidate(a).
3
+ candidate(b).
4
+ allowed(a).
5
+ answer(X, ok) :- candidate(X), allowed(X).
6
+
7
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 5.4: improper list surface syntax unifies with head-tail structure.
2
+ cell([head | tail], head, tail).
3
+ answer(list, L) :- cell(L, head, tail).
4
+ answer(head, H) :- cell([H | tail], H, tail).
5
+ answer(tail, T) :- cell([head | T], head, T).
6
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 4, 5.3: zero-arity compounds are written and matched with parentheses.
2
+ status(nil(), ok).
3
+ answer(value, X) :- status(X, ok).
4
+ materialize(answer, 2).
@@ -0,0 +1,8 @@
1
+ % Reference 7: recursive definite clauses can derive paths beyond one step.
2
+ link(a, b).
3
+ link(b, c).
4
+ link(c, d).
5
+ path(X, Y) :- link(X, Y).
6
+ path(X, Z) :- link(X, Y), path(Y, Z).
7
+ answer(reachable, X) :- path(a, X).
8
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 3.5, 11: quoted atoms preserve spaces, quotes, and the empty atom.
2
+ symbol('two words').
3
+ symbol('needs''quote').
4
+ symbol('').
5
+ answer(symbol, X) :- symbol(X).
6
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 5.5, 7: parenthesized comma terms with more than two members are conjunctions as goals.
2
+ p(a).
3
+ q(a).
4
+ r(a).
5
+ ok(X) :- (p(X), q(X), r(X)).
6
+ answer(ok, X) :- ok(X).
7
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 5.3, 5.4: lists may contain structured terms that unify positionally.
2
+ node([pair(a, b), pair(c, d)]).
3
+ answer(first_key, X) :- node([pair(X, _), pair(c, d)]).
4
+ answer(second_key, X) :- node([pair(a, b), pair(X, d)]).
5
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 5.1, 7.1: repeated variables in a rule body require the same term.
2
+ pair(a, a).
3
+ pair(a, b).
4
+ pair(c, c).
5
+ diagonal(X) :- pair(X, X).
6
+ answer(diagonal, X) :- diagonal(X).
7
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 5.3, 6, 7.1: structured rule heads destructure matching goals.
2
+ unpack(pair(X, Y), X, Y).
3
+ answer(first, A) :- unpack(pair(alpha, beta), A, _).
4
+ answer(second, B) :- unpack(pair(alpha, beta), _, B).
5
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 3.5, 11: quoted strings and atoms preserve escape sequences at read-back.
2
+ raw(string, "line\nnext\t\\slash").
3
+ raw(atom, 'line\nnext\t\\slash').
4
+ answer(K, V) :- raw(K, V).
5
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 3.3, 11: signed, decimal, and exponent numeric literals retain lexical form.
2
+ raw(negative_decimal, -3.5).
3
+ raw(positive_exp, 6.02e23).
4
+ raw(negative_exp, -1.0e-3).
5
+ answer(K, V) :- raw(K, V).
6
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 5.5, 7: formula data can be passed through parenthesized conjunctions.
2
+ record((left(a), right(b))).
3
+ accept((left(a), right(b))).
4
+ answer(ok, F) :- (record(F), accept(F)).
5
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 3.4, 5.1: variables beginning with underscore are named variables unless exactly _.
2
+ pair(a, a).
3
+ pair(a, b).
4
+ answer(shared, _Value) :- pair(_Value, _Value).
5
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 11: default output prints new ground binary derivations, not source facts.
2
+ parent(pat, jan).
3
+ parent(jan, emma).
4
+ ancestor(X, Y) :- parent(X, Y).
5
+ ancestor(X, Z) :- parent(X, Y), ancestor(Y, Z).
@@ -0,0 +1,5 @@
1
+ % Reference 10.2, 11: materialize/2 restricts selected default predicate groups.
2
+ materialize(answer, 2).
3
+ seed(a).
4
+ helper(X, y) :- seed(X).
5
+ answer(X, ok) :- helper(X, y).
@@ -0,0 +1,11 @@
1
+ % Reference 9.2, 9.3: arithmetic and comparison built-ins.
2
+ answer(sum, X) :- add(2, 3, X).
3
+ answer(diff, X) :- sub(7, 4, X).
4
+ answer(product, X) :- mul(6, 7, X).
5
+ answer(integer_division, X) :- div(7, 2, X).
6
+ answer(remainder, X) :- mod(7, 2, X).
7
+ answer(power, X) :- pow(2, 8, X).
8
+ answer(minimum, X) :- min(3, 9, X).
9
+ answer(less_than, true) :- lt(3, 9).
10
+ answer(greater_equal, true) :- ge(9, 9).
11
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 9.6: atom and string built-ins.
2
+ answer(str_concat, X) :- str_concat("eye", "lang", X).
3
+ answer(contains, true) :- contains("eyelang", "lang").
4
+ materialize(answer, 2).
@@ -0,0 +1,10 @@
1
+ % Reference 9.7, 9.8: list, aggregation, and ordering built-ins.
2
+ answer(member, X) :- member(X, [a, b]).
3
+ answer(append, X) :- append([a], [b, c], X).
4
+ answer(nth0, X) :- nth0(1, [a, b, c], X).
5
+ answer(set_nth0, X) :- set_nth0(1, [a, b, c], x, X).
6
+ answer(reverse, X) :- reverse([a, b, c], X).
7
+ answer(length, N) :- length([a, b, c], N).
8
+ answer(findall, X) :- findall(N, between(1, 3, N), X).
9
+ answer(sort, X) :- sort([b, a, b], X).
10
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 9.9: holds/3 exposes context members of any arity.
2
+ context((ready, name(alice, "Alice"), route(alice, bob, 7))).
3
+ answer(parts, exposed(Name, Args)) :- context(C), holds(C, Name, Args).
4
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 9.5, 9.10: generators, negation as failure, and once/1.
2
+ candidate(a).
3
+ candidate(b).
4
+ closed(b).
5
+ answer(open, X) :- candidate(X), not(closed(X)).
6
+ answer(first_between, X) :- once(between(2, 4, X)).
7
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 9.1: eq/2 unifies terms and neq/2 succeeds on non-unifiable terms.
2
+ answer(eq_variable, X) :- eq(X, pair(a, [b, c])).
3
+ answer(eq_nested, true) :- eq(pair(X, X), pair(same, same)).
4
+ answer(neq_atoms, true) :- neq(alice, bob).
5
+ answer(neq_structures, true) :- neq(pair(a), pair(a, b)).
6
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 9.7: rest/2, select/3, and not_member/2.
2
+ answer(rest, X) :- rest([a, b, c], X).
3
+ answer(select, selected(X, R)) :- select(X, [a, b], R).
4
+ answer(not_member, true) :- not_member(c, [a, b]).
5
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.7: append/3 can enumerate proper prefix/suffix splits.
2
+ answer(split, split(A, B)) :- append(A, B, [a, b]).
3
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 9.3, 9.6: lexical comparison and simple text matching.
2
+ answer(matches, true) :- matches("eyelang", "eye").
3
+ answer(not_matches, true) :- not_matches("eyelang", "cat").
4
+ answer(lex_lt, true) :- lt(alpha, beta).
5
+ answer(lex_gt, true) :- gt(beta, alpha).
6
+ answer(numeric_le, true) :- le(2, 2).
7
+ materialize(answer, 2).
@@ -0,0 +1,8 @@
1
+ % Reference 10.1: memoize/2 is a declaration and does not change answers.
2
+ materialize(reach, 2).
3
+ memoize(reach_any, 2).
4
+ edge(a, b).
5
+ edge(b, c).
6
+ reach_any(X, Y) :- edge(X, Y).
7
+ reach_any(X, Z) :- edge(X, Y), reach_any(Y, Z).
8
+ reach(a, Y) :- reach_any(a, Y).
@@ -0,0 +1,9 @@
1
+ % Reference 9.2: scalar arithmetic functions return numeric values.
2
+ answer(neg, X) :- neg(5, X).
3
+ answer(abs, X) :- abs(-5, X).
4
+ answer(rounded, X) :- rounded(2.6, X).
5
+ answer(sin_zero, X) :- sin(0, X).
6
+ answer(cos_zero, X) :- cos(0, X).
7
+ answer(log_one, X) :- log(1, X).
8
+ answer(float_division, X) :- div(7.0, 2.0, X).
9
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.5: between/3 enumerates every integer in an inclusive range.
2
+ answer(n, X) :- between(3, 5, X).
3
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.5: smallest_divisor_from/3 finds the first divisor at or above the start.
2
+ answer(divisor, D) :- smallest_divisor_from(21, 2, D).
3
+ materialize(answer, 2).
@@ -0,0 +1,7 @@
1
+ % Reference 9.10: not/1 succeeds when the subgoal has no proof.
2
+ candidate(a).
3
+ candidate(b).
4
+ candidate(c).
5
+ blocked(b).
6
+ answer(open, X) :- candidate(X), not(blocked(X)).
7
+ materialize(answer, 2).
@@ -0,0 +1,5 @@
1
+ % Reference 9.10: once/1 keeps at most the first solution from a user predicate.
2
+ choice(a).
3
+ choice(b).
4
+ answer(first, X) :- once(choice(X)).
5
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 9.8: findall/3 collects templates for every solution of a user goal.
2
+ p(b).
3
+ p(a).
4
+ p(b).
5
+ answer(bag, X) :- findall(P, p(P), X).
6
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.8: sort/2 sorts and deduplicates a proper list.
2
+ answer(sorted, X) :- sort([c, a, b, a], X).
3
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 9.7: append/3 supports bound prefix and suffix use cases.
2
+ answer(prefix, X) :- append(X, [c], [a, b, c]).
3
+ answer(suffix, X) :- append([a], X, [a, b, c]).
4
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.7: nth0/3 can bind the index for a known list element.
2
+ answer(index, I) :- nth0(I, [a, b, c], b).
3
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 9.7: set_nth0/4 updates zero-based positions functionally.
2
+ answer(first, X) :- set_nth0(0, [a, b, c], x, X).
3
+ answer(last, X) :- set_nth0(2, [a, b, c], z, X).
4
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.7: select/3 enumerates removals of matching occurrences.
2
+ answer(rest, X) :- select(a, [a, b, a], X).
3
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 9.7: not_member/2 can filter finite candidates.
2
+ candidate(a).
3
+ candidate(b).
4
+ candidate(c).
5
+ answer(not_present, X) :- candidate(X), not_member(X, [a, b]).
6
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ % Reference 9.9: holds/3 enumerates nested comma contexts.
2
+ context(((ready, name(a, "A")), route(a, b, 7))).
3
+ answer(parts, exposed(Name, Args)) :- context(C), holds(C, Name, Args).
4
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.9: holds/2 over comma context data.
2
+ context((name(alice, "Alice"), knows(alice, bob))).
3
+ answer(member, Term) :- context(C), holds(C, Term).
@@ -0,0 +1,6 @@
1
+ % Reference 11: default materialization excludes source facts even if also derivable.
2
+ materialize(answer, 2).
3
+ seed(a).
4
+ answer(a, ok).
5
+ answer(X, ok) :- seed(X).
6
+ answer(b, ok) :- seed(a).
@@ -0,0 +1,5 @@
1
+ % Reference 9.3: comparisons handle numeric and lexical scalar ordering.
2
+ answer(numeric_gt, true) :- gt(10, 2).
3
+ answer(numeric_le, true) :- le(2, 2.0).
4
+ answer(lexical_ge, true) :- ge(beta, alpha).
5
+ materialize(answer, 2).
@@ -0,0 +1,6 @@
1
+ % Reference 9.6: matches/2 and not_matches/2 can filter candidate strings.
2
+ text(a, "alpha").
3
+ text(b, "beta").
4
+ answer(has_ph, K) :- text(K, T), matches(T, "ph").
5
+ answer(no_ph, K) :- text(K, T), not_matches(T, "ph").
6
+ materialize(answer, 2).
@@ -0,0 +1,3 @@
1
+ % Reference 9.6: str_concat/3 concatenates strings.
2
+ answer(string, X) :- str_concat("eye", "lang", X).
3
+ materialize(answer, 2).
@@ -0,0 +1,4 @@
1
+ item(a).
2
+ item(b).
3
+ answer(counts, counts(C, Z)) :- countall(item(X), C), countall(missing(X), Z).
4
+ materialize(answer, 2).
@@ -0,0 +1 @@
1
+ (countall(item(X), C), countall(missing(X), Z))
@@ -0,0 +1,5 @@
1
+ score(a, 4).
2
+ score(b, 5).
3
+ score(c, -2).
4
+ answer(sum, Sum) :- sumall(S, score(_Item, S), Sum).
5
+ materialize(answer, 2).
@@ -0,0 +1 @@
1
+ sumall(S, score(_Item, S), Sum)
@@ -0,0 +1,5 @@
1
+ score(alpha, 7).
2
+ score(beta, 3).
3
+ score(gamma, 5).
4
+ answer(min, result(BestS, Best)) :- aggregate_min(S, item(Name, S), score(Name, S), BestS, Best).
5
+ materialize(answer, 2).
@@ -0,0 +1 @@
1
+ aggregate_min(S, item(Name, S), score(Name, S), BestS, Best)
@@ -0,0 +1,5 @@
1
+ score(alpha, 7).
2
+ score(beta, 7).
3
+ score(gamma, 5).
4
+ answer(max, result(Key, BestName)) :- aggregate_max([S, Name], Name, score(Name, S), Key, BestName).
5
+ materialize(answer, 2).
@@ -0,0 +1 @@
1
+ aggregate_max([S, Name], Name, score(Name, S), Key, BestName)
@@ -0,0 +1,4 @@
1
+ % Reference 9.1: difference/3 computes ISO date durations.
2
+ answer(duration, D) :- difference("2024-03-01", "2020-02-29", D).
3
+ answer(month_borrow, D) :- difference("2024-03-01", "2024-01-31", D).
4
+ materialize(answer, 2).
@@ -0,0 +1,10 @@
1
+ % Reference 9.1: reusable numeric functions and max/3.
2
+ materialize(answer, 2).
3
+ answer(max, X) :- max(17, 42, X).
4
+ answer(sqrt, X) :- sqrt(81, X).
5
+ answer(floor, X) :- floor(3.9, X).
6
+ answer(ceiling, X) :- ceiling(3.1, X).
7
+ answer(trunc, X) :- trunc(-3.9, X).
8
+ answer(exp, X) :- exp(0, X).
9
+ answer(tan, X) :- tan(0, X).
10
+ answer(atan2, X) :- atan2(0, -1, X).
@@ -0,0 +1,11 @@
1
+ % Reference 9.1: reusable list selectors, slices, summaries, and stable set conversion.
2
+ materialize(answer, 2).
3
+ answer(head, X) :- head([alpha, beta, gamma, beta], X).
4
+ answer(last, X) :- last([alpha, beta, gamma, beta], X).
5
+ answer(take, X) :- take(2, [alpha, beta, gamma, beta], X).
6
+ answer(drop, X) :- drop(2, [alpha, beta, gamma, beta], X).
7
+ answer(slice, X) :- slice(1, 2, [alpha, beta, gamma, beta], X).
8
+ answer(sum, X) :- sum_list([1, 2, 3.5], X).
9
+ answer(min, X) :- min_list([3, 1, 2], X).
10
+ answer(max, X) :- max_list([3, 1, 2], X).
11
+ answer(set, X) :- list_to_set([beta, alpha, beta, gamma, alpha], X).
@@ -0,0 +1,12 @@
1
+ % Reference 9.1: reusable string normalization, splitting, joining, replacement, and conversions.
2
+ materialize(answer, 2).
3
+ answer(trim, X) :- trim(" Hello Eyelang ", X).
4
+ answer(lower, X) :- lowercase("Hello Eyelang", X).
5
+ answer(upper, X) :- uppercase("Hello Eyelang", X).
6
+ answer(split, X) :- split("red,green,blue", ",", X).
7
+ answer(join, X) :- join([red, green, blue], ":", X).
8
+ answer(substring, X) :- substring("abcdef", 2, 3, X).
9
+ answer(replace, X) :- replace("red-green-red", "red", "blue", X).
10
+ answer(number_to_string, X) :- number_string(42, X).
11
+ answer(string_to_number, X) :- number_string(X, "3.5").
12
+ answer(atom_string, X) :- atom_string(eyelang, X).
@@ -0,0 +1,11 @@
1
+ % Reference 9.1: term introspection/construction, term strings, and forall/2.
2
+ materialize(answer, 2).
3
+ item(1).
4
+ item(2).
5
+ item(3).
6
+ answer(functor, pair(Name, Arity)) :- functor(edge(a, b), Name, Arity).
7
+ answer(arg, X) :- arg(2, edge(a, b), X).
8
+ answer(decompose, pair(Name, Args)) :- compound_name_arguments(edge(a, b), Name, Args).
9
+ answer(compose, X) :- compound_name_arguments(X, edge, [a, b]).
10
+ answer(term_string, X) :- term_string(edge(a, [b, c]), X).
11
+ answer(forall, ok) :- forall(item(X), lt(X, 4)).
@@ -0,0 +1,14 @@
1
+ % Reference 9.1: reusable numeric functions preserve integer paths and define finite failure modes.
2
+ materialize(answer, 2).
3
+ answer(max_negative, X) :- max(-10, -3, X).
4
+ answer(min_float, X) :- min(2.5, -1.25, X).
5
+ answer(floor_negative, X) :- floor(-3.1, X).
6
+ answer(ceiling_negative, X) :- ceiling(-3.9, X).
7
+ answer(trunc_positive, X) :- trunc(3.9, X).
8
+ answer(sqrt_fraction, X) :- sqrt(2.25, X).
9
+ answer(pow_fraction, X) :- pow(9, 0.5, X).
10
+ answer(sqrt_negative_rejected, ok) :- not(sqrt(-1, X)).
11
+ answer(log_zero_rejected, ok) :- not(log(0, X)).
12
+ answer(div_zero_rejected, ok) :- not(div(1, 0, X)).
13
+ answer(mod_float_rejected, ok) :- not(mod(5.5, 2, X)).
14
+ answer(pow_negative_integer_exponent_rejected, ok) :- not(pow(2, -1, X)).
@@ -0,0 +1,10 @@
1
+ % Reference 9.1: reusable list selectors and slices have explicit finite boundary behavior.
2
+ materialize(answer, 2).
3
+ answer(take_zero, X) :- take(0, [a, b, c], X).
4
+ answer(drop_all, X) :- drop(3, [a, b, c], X).
5
+ answer(slice_empty, X) :- slice(1, 0, [a, b, c], X).
6
+ answer(last_single, X) :- last([only], X).
7
+ answer(head_rest, pair(H, R)) :- head([a, b, c], H), rest([a, b, c], R).
8
+ answer(take_too_many_rejected, ok) :- not(take(4, [a, b, c], X)).
9
+ answer(drop_too_many_rejected, ok) :- not(drop(4, [a, b, c], X)).
10
+ answer(last_empty_rejected, ok) :- not(last([], X)).
@@ -0,0 +1,7 @@
1
+ % Reference 9.1: reusable list relations enumerate in stable left-to-right order.
2
+ materialize(answer, 2).
3
+ answer(append_split, pair(Prefix, Suffix)) :- append(Prefix, Suffix, [a, b]).
4
+ answer(nth, pair(Index, Value)) :- nth0(Index, [x, y], Value).
5
+ answer(select, pair(Value, Rest)) :- select(Value, [a, b, a], Rest).
6
+ answer(not_member_atom, ok) :- not_member(z, [a, b, c]).
7
+ answer(not_member_unifiable_rejected, ok) :- not(not_member(pair(X), [pair(a)])).
@@ -0,0 +1,9 @@
1
+ % Reference 9.1: list summaries accept mixed numeric values and stable duplicate removal.
2
+ materialize(answer, 2).
3
+ answer(sum_empty, X) :- sum_list([], X).
4
+ answer(sum_mixed, X) :- sum_list([1, 2.5, 3], X).
5
+ answer(min_atom_order, X) :- min_list([delta, beta, gamma], X).
6
+ answer(max_atom_order, X) :- max_list([delta, beta, gamma], X).
7
+ answer(set_terms, X) :- list_to_set([pair(a, 1), pair(a, 1), pair(b, 2), pair(a, 1)], X).
8
+ answer(min_empty_rejected, ok) :- not(min_list([], X)).
9
+ answer(max_empty_rejected, ok) :- not(max_list([], X)).
@@ -0,0 +1,13 @@
1
+ % Reference 9.6: matches/3 extracts named regular-expression captures into a context.
2
+ materialize(answer, 3).
3
+
4
+ line("event=login_failed user=alice trace=4bf92f3577b34da6a3ce929d0e0e4736").
5
+ pattern("^event=(?<event>\\w+) user=(?<user>\\w+) trace=(?<trace_id>[0-9a-f]{32})$").
6
+
7
+ answer(User, Event, TraceId) :-
8
+ line(Text),
9
+ pattern(Pattern),
10
+ matches(Text, Pattern, Context),
11
+ holds(Context, event(Event)),
12
+ holds(Context, user(User)),
13
+ holds(Context, trace_id(TraceId)).