relationalai 0.12.13__py3-none-any.whl → 0.13.0.dev0__py3-none-any.whl

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 (825) hide show
  1. relationalai/__init__.py +1 -209
  2. relationalai/config/__init__.py +56 -0
  3. relationalai/config/config.py +289 -0
  4. relationalai/config/config_fields.py +86 -0
  5. relationalai/config/connections/__init__.py +46 -0
  6. relationalai/config/connections/base.py +23 -0
  7. relationalai/config/connections/duckdb.py +29 -0
  8. relationalai/config/connections/snowflake.py +243 -0
  9. relationalai/config/external/__init__.py +17 -0
  10. relationalai/config/external/dbt_converter.py +101 -0
  11. relationalai/config/external/dbt_models.py +93 -0
  12. relationalai/config/external/snowflake_converter.py +41 -0
  13. relationalai/config/external/snowflake_models.py +85 -0
  14. relationalai/config/external/utils.py +19 -0
  15. relationalai/semantics/__init__.py +146 -22
  16. relationalai/semantics/backends/lqp/annotations.py +11 -0
  17. relationalai/semantics/backends/sql/sql_compiler.py +327 -0
  18. relationalai/semantics/frontend/base.py +1707 -0
  19. relationalai/semantics/frontend/core.py +179 -0
  20. relationalai/semantics/frontend/front_compiler.py +1313 -0
  21. relationalai/semantics/frontend/pprint.py +408 -0
  22. relationalai/semantics/metamodel/__init__.py +6 -40
  23. relationalai/semantics/metamodel/builtins.py +205 -769
  24. relationalai/semantics/metamodel/metamodel.py +437 -0
  25. relationalai/semantics/metamodel/metamodel_analyzer.py +519 -0
  26. relationalai/semantics/metamodel/pprint.py +412 -0
  27. relationalai/semantics/metamodel/rewriter.py +266 -0
  28. relationalai/semantics/metamodel/typer.py +1378 -0
  29. relationalai/semantics/std/__init__.py +60 -40
  30. relationalai/semantics/std/aggregates.py +149 -0
  31. relationalai/semantics/std/common.py +44 -0
  32. relationalai/semantics/std/constraints.py +37 -43
  33. relationalai/semantics/std/datetime.py +246 -135
  34. relationalai/semantics/std/decimals.py +45 -52
  35. relationalai/semantics/std/floats.py +13 -5
  36. relationalai/semantics/std/integers.py +26 -11
  37. relationalai/semantics/std/math.py +183 -112
  38. relationalai/semantics/std/numbers.py +86 -0
  39. relationalai/semantics/std/re.py +80 -62
  40. relationalai/semantics/std/strings.py +117 -60
  41. relationalai/shims/executor.py +147 -0
  42. relationalai/shims/helpers.py +126 -0
  43. relationalai/shims/hoister.py +221 -0
  44. relationalai/shims/mm2v0.py +1290 -0
  45. relationalai/tools/cli/__init__.py +6 -0
  46. relationalai/tools/cli/cli.py +90 -0
  47. relationalai/tools/cli/components/__init__.py +5 -0
  48. relationalai/tools/cli/components/progress_reader.py +1524 -0
  49. relationalai/tools/cli/components/utils.py +58 -0
  50. relationalai/tools/cli/config_template.py +45 -0
  51. relationalai/tools/cli/dev.py +19 -0
  52. relationalai/tools/debugger.py +289 -183
  53. relationalai/tools/typer_debugger.py +93 -0
  54. relationalai/util/dataclasses.py +43 -0
  55. relationalai/util/docutils.py +40 -0
  56. relationalai/util/error.py +199 -0
  57. relationalai/util/format.py +48 -106
  58. relationalai/util/naming.py +145 -0
  59. relationalai/util/python.py +35 -0
  60. relationalai/util/runtime.py +156 -0
  61. relationalai/util/schema.py +197 -0
  62. relationalai/util/source.py +185 -0
  63. relationalai/util/structures.py +163 -0
  64. relationalai/util/tracing.py +261 -0
  65. relationalai-0.13.0.dev0.dist-info/METADATA +46 -0
  66. relationalai-0.13.0.dev0.dist-info/RECORD +488 -0
  67. relationalai-0.13.0.dev0.dist-info/WHEEL +5 -0
  68. relationalai-0.13.0.dev0.dist-info/entry_points.txt +3 -0
  69. relationalai-0.13.0.dev0.dist-info/top_level.txt +2 -0
  70. v0/relationalai/__init__.py +216 -0
  71. v0/relationalai/clients/azure.py +477 -0
  72. v0/relationalai/clients/client.py +912 -0
  73. v0/relationalai/clients/config.py +673 -0
  74. v0/relationalai/clients/direct_access_client.py +118 -0
  75. v0/relationalai/clients/hash_util.py +31 -0
  76. v0/relationalai/clients/local.py +571 -0
  77. v0/relationalai/clients/profile_polling.py +73 -0
  78. v0/relationalai/clients/result_helpers.py +420 -0
  79. v0/relationalai/clients/snowflake.py +3869 -0
  80. v0/relationalai/clients/types.py +113 -0
  81. v0/relationalai/clients/use_index_poller.py +980 -0
  82. v0/relationalai/clients/util.py +356 -0
  83. v0/relationalai/debugging.py +389 -0
  84. v0/relationalai/dsl.py +1749 -0
  85. v0/relationalai/early_access/builder/__init__.py +30 -0
  86. v0/relationalai/early_access/builder/builder/__init__.py +35 -0
  87. v0/relationalai/early_access/builder/snowflake/__init__.py +12 -0
  88. v0/relationalai/early_access/builder/std/__init__.py +25 -0
  89. v0/relationalai/early_access/builder/std/decimals/__init__.py +12 -0
  90. v0/relationalai/early_access/builder/std/integers/__init__.py +12 -0
  91. v0/relationalai/early_access/builder/std/math/__init__.py +12 -0
  92. v0/relationalai/early_access/builder/std/strings/__init__.py +14 -0
  93. v0/relationalai/early_access/devtools/__init__.py +12 -0
  94. v0/relationalai/early_access/devtools/benchmark_lqp/__init__.py +12 -0
  95. v0/relationalai/early_access/devtools/extract_lqp/__init__.py +12 -0
  96. v0/relationalai/early_access/dsl/adapters/orm/adapter_qb.py +427 -0
  97. v0/relationalai/early_access/dsl/adapters/orm/parser.py +636 -0
  98. v0/relationalai/early_access/dsl/adapters/owl/adapter.py +176 -0
  99. v0/relationalai/early_access/dsl/adapters/owl/parser.py +160 -0
  100. v0/relationalai/early_access/dsl/bindings/common.py +402 -0
  101. v0/relationalai/early_access/dsl/bindings/csv.py +170 -0
  102. v0/relationalai/early_access/dsl/bindings/legacy/binding_models.py +143 -0
  103. v0/relationalai/early_access/dsl/bindings/snowflake.py +64 -0
  104. v0/relationalai/early_access/dsl/codegen/binder.py +411 -0
  105. v0/relationalai/early_access/dsl/codegen/common.py +79 -0
  106. v0/relationalai/early_access/dsl/codegen/helpers.py +23 -0
  107. v0/relationalai/early_access/dsl/codegen/relations.py +700 -0
  108. v0/relationalai/early_access/dsl/codegen/weaver.py +417 -0
  109. v0/relationalai/early_access/dsl/core/builders/__init__.py +47 -0
  110. v0/relationalai/early_access/dsl/core/builders/logic.py +19 -0
  111. v0/relationalai/early_access/dsl/core/builders/scalar_constraint.py +11 -0
  112. v0/relationalai/early_access/dsl/core/constraints/predicate/atomic.py +455 -0
  113. v0/relationalai/early_access/dsl/core/constraints/predicate/universal.py +73 -0
  114. v0/relationalai/early_access/dsl/core/constraints/scalar.py +310 -0
  115. v0/relationalai/early_access/dsl/core/context.py +13 -0
  116. v0/relationalai/early_access/dsl/core/cset.py +132 -0
  117. v0/relationalai/early_access/dsl/core/exprs/__init__.py +116 -0
  118. v0/relationalai/early_access/dsl/core/exprs/relational.py +18 -0
  119. v0/relationalai/early_access/dsl/core/exprs/scalar.py +412 -0
  120. v0/relationalai/early_access/dsl/core/instances.py +44 -0
  121. v0/relationalai/early_access/dsl/core/logic/__init__.py +193 -0
  122. v0/relationalai/early_access/dsl/core/logic/aggregation.py +98 -0
  123. v0/relationalai/early_access/dsl/core/logic/exists.py +223 -0
  124. v0/relationalai/early_access/dsl/core/logic/helper.py +163 -0
  125. v0/relationalai/early_access/dsl/core/namespaces.py +32 -0
  126. v0/relationalai/early_access/dsl/core/relations.py +276 -0
  127. v0/relationalai/early_access/dsl/core/rules.py +112 -0
  128. v0/relationalai/early_access/dsl/core/std/__init__.py +45 -0
  129. v0/relationalai/early_access/dsl/core/temporal/recall.py +6 -0
  130. v0/relationalai/early_access/dsl/core/types/__init__.py +270 -0
  131. v0/relationalai/early_access/dsl/core/types/concepts.py +128 -0
  132. v0/relationalai/early_access/dsl/core/types/constrained/__init__.py +267 -0
  133. v0/relationalai/early_access/dsl/core/types/constrained/nominal.py +143 -0
  134. v0/relationalai/early_access/dsl/core/types/constrained/subtype.py +124 -0
  135. v0/relationalai/early_access/dsl/core/types/standard.py +92 -0
  136. v0/relationalai/early_access/dsl/core/types/unconstrained.py +50 -0
  137. v0/relationalai/early_access/dsl/core/types/variables.py +203 -0
  138. v0/relationalai/early_access/dsl/ir/compiler.py +318 -0
  139. v0/relationalai/early_access/dsl/ir/executor.py +260 -0
  140. v0/relationalai/early_access/dsl/ontologies/constraints.py +88 -0
  141. v0/relationalai/early_access/dsl/ontologies/export.py +30 -0
  142. v0/relationalai/early_access/dsl/ontologies/models.py +453 -0
  143. v0/relationalai/early_access/dsl/ontologies/python_printer.py +303 -0
  144. v0/relationalai/early_access/dsl/ontologies/readings.py +60 -0
  145. v0/relationalai/early_access/dsl/ontologies/relationships.py +322 -0
  146. v0/relationalai/early_access/dsl/ontologies/roles.py +87 -0
  147. v0/relationalai/early_access/dsl/ontologies/subtyping.py +55 -0
  148. v0/relationalai/early_access/dsl/orm/constraints.py +438 -0
  149. v0/relationalai/early_access/dsl/orm/measures/dimensions.py +200 -0
  150. v0/relationalai/early_access/dsl/orm/measures/initializer.py +16 -0
  151. v0/relationalai/early_access/dsl/orm/measures/measure_rules.py +275 -0
  152. v0/relationalai/early_access/dsl/orm/measures/measures.py +299 -0
  153. v0/relationalai/early_access/dsl/orm/measures/role_exprs.py +268 -0
  154. v0/relationalai/early_access/dsl/orm/models.py +256 -0
  155. v0/relationalai/early_access/dsl/orm/object_oriented_printer.py +344 -0
  156. v0/relationalai/early_access/dsl/orm/printer.py +469 -0
  157. v0/relationalai/early_access/dsl/orm/reasoners.py +480 -0
  158. v0/relationalai/early_access/dsl/orm/relations.py +19 -0
  159. v0/relationalai/early_access/dsl/orm/relationships.py +251 -0
  160. v0/relationalai/early_access/dsl/orm/types.py +42 -0
  161. v0/relationalai/early_access/dsl/orm/utils.py +79 -0
  162. v0/relationalai/early_access/dsl/orm/verb.py +204 -0
  163. v0/relationalai/early_access/dsl/physical_metadata/tables.py +133 -0
  164. v0/relationalai/early_access/dsl/relations.py +170 -0
  165. v0/relationalai/early_access/dsl/rulesets.py +69 -0
  166. v0/relationalai/early_access/dsl/schemas/__init__.py +450 -0
  167. v0/relationalai/early_access/dsl/schemas/builder.py +48 -0
  168. v0/relationalai/early_access/dsl/schemas/comp_names.py +51 -0
  169. v0/relationalai/early_access/dsl/schemas/components.py +203 -0
  170. v0/relationalai/early_access/dsl/schemas/contexts.py +156 -0
  171. v0/relationalai/early_access/dsl/schemas/exprs.py +89 -0
  172. v0/relationalai/early_access/dsl/schemas/fragments.py +464 -0
  173. v0/relationalai/early_access/dsl/serialization.py +79 -0
  174. v0/relationalai/early_access/dsl/serialize/exporter.py +163 -0
  175. v0/relationalai/early_access/dsl/snow/api.py +104 -0
  176. v0/relationalai/early_access/dsl/snow/common.py +76 -0
  177. v0/relationalai/early_access/dsl/state_mgmt/__init__.py +129 -0
  178. v0/relationalai/early_access/dsl/state_mgmt/state_charts.py +125 -0
  179. v0/relationalai/early_access/dsl/state_mgmt/transitions.py +130 -0
  180. v0/relationalai/early_access/dsl/types/__init__.py +40 -0
  181. v0/relationalai/early_access/dsl/types/concepts.py +12 -0
  182. v0/relationalai/early_access/dsl/types/entities.py +135 -0
  183. v0/relationalai/early_access/dsl/types/values.py +17 -0
  184. v0/relationalai/early_access/dsl/utils.py +102 -0
  185. v0/relationalai/early_access/graphs/__init__.py +13 -0
  186. v0/relationalai/early_access/lqp/__init__.py +12 -0
  187. v0/relationalai/early_access/lqp/compiler/__init__.py +12 -0
  188. v0/relationalai/early_access/lqp/constructors/__init__.py +18 -0
  189. v0/relationalai/early_access/lqp/executor/__init__.py +12 -0
  190. v0/relationalai/early_access/lqp/ir/__init__.py +12 -0
  191. v0/relationalai/early_access/lqp/passes/__init__.py +12 -0
  192. v0/relationalai/early_access/lqp/pragmas/__init__.py +12 -0
  193. v0/relationalai/early_access/lqp/primitives/__init__.py +12 -0
  194. v0/relationalai/early_access/lqp/types/__init__.py +12 -0
  195. v0/relationalai/early_access/lqp/utils/__init__.py +12 -0
  196. v0/relationalai/early_access/lqp/validators/__init__.py +12 -0
  197. v0/relationalai/early_access/metamodel/__init__.py +58 -0
  198. v0/relationalai/early_access/metamodel/builtins/__init__.py +12 -0
  199. v0/relationalai/early_access/metamodel/compiler/__init__.py +12 -0
  200. v0/relationalai/early_access/metamodel/dependency/__init__.py +12 -0
  201. v0/relationalai/early_access/metamodel/factory/__init__.py +17 -0
  202. v0/relationalai/early_access/metamodel/helpers/__init__.py +12 -0
  203. v0/relationalai/early_access/metamodel/ir/__init__.py +14 -0
  204. v0/relationalai/early_access/metamodel/rewrite/__init__.py +7 -0
  205. v0/relationalai/early_access/metamodel/typer/__init__.py +3 -0
  206. v0/relationalai/early_access/metamodel/typer/typer/__init__.py +12 -0
  207. v0/relationalai/early_access/metamodel/types/__init__.py +15 -0
  208. v0/relationalai/early_access/metamodel/util/__init__.py +15 -0
  209. v0/relationalai/early_access/metamodel/visitor/__init__.py +12 -0
  210. v0/relationalai/early_access/rel/__init__.py +12 -0
  211. v0/relationalai/early_access/rel/executor/__init__.py +12 -0
  212. v0/relationalai/early_access/rel/rel_utils/__init__.py +12 -0
  213. v0/relationalai/early_access/rel/rewrite/__init__.py +7 -0
  214. v0/relationalai/early_access/solvers/__init__.py +19 -0
  215. v0/relationalai/early_access/sql/__init__.py +11 -0
  216. v0/relationalai/early_access/sql/executor/__init__.py +3 -0
  217. v0/relationalai/early_access/sql/rewrite/__init__.py +3 -0
  218. v0/relationalai/early_access/tests/logging/__init__.py +12 -0
  219. v0/relationalai/early_access/tests/test_snapshot_base/__init__.py +12 -0
  220. v0/relationalai/early_access/tests/utils/__init__.py +12 -0
  221. v0/relationalai/environments/__init__.py +35 -0
  222. v0/relationalai/environments/base.py +381 -0
  223. v0/relationalai/environments/colab.py +14 -0
  224. v0/relationalai/environments/generic.py +71 -0
  225. v0/relationalai/environments/ipython.py +68 -0
  226. v0/relationalai/environments/jupyter.py +9 -0
  227. v0/relationalai/environments/snowbook.py +169 -0
  228. v0/relationalai/errors.py +2455 -0
  229. v0/relationalai/experimental/SF.py +38 -0
  230. v0/relationalai/experimental/inspect.py +47 -0
  231. v0/relationalai/experimental/pathfinder/__init__.py +158 -0
  232. v0/relationalai/experimental/pathfinder/api.py +160 -0
  233. v0/relationalai/experimental/pathfinder/automaton.py +584 -0
  234. v0/relationalai/experimental/pathfinder/bridge.py +226 -0
  235. v0/relationalai/experimental/pathfinder/compiler.py +416 -0
  236. v0/relationalai/experimental/pathfinder/datalog.py +214 -0
  237. v0/relationalai/experimental/pathfinder/diagnostics.py +56 -0
  238. v0/relationalai/experimental/pathfinder/filter.py +236 -0
  239. v0/relationalai/experimental/pathfinder/glushkov.py +439 -0
  240. v0/relationalai/experimental/pathfinder/options.py +265 -0
  241. v0/relationalai/experimental/pathfinder/rpq.py +344 -0
  242. v0/relationalai/experimental/pathfinder/transition.py +200 -0
  243. v0/relationalai/experimental/pathfinder/utils.py +26 -0
  244. v0/relationalai/experimental/paths/api.py +143 -0
  245. v0/relationalai/experimental/paths/benchmarks/grid_graph.py +37 -0
  246. v0/relationalai/experimental/paths/examples/basic_example.py +40 -0
  247. v0/relationalai/experimental/paths/examples/minimal_engine_warmup.py +3 -0
  248. v0/relationalai/experimental/paths/examples/movie_example.py +77 -0
  249. v0/relationalai/experimental/paths/examples/paths_benchmark.py +115 -0
  250. v0/relationalai/experimental/paths/examples/paths_example.py +116 -0
  251. v0/relationalai/experimental/paths/examples/pattern_to_automaton.py +28 -0
  252. v0/relationalai/experimental/paths/find_paths_via_automaton.py +85 -0
  253. v0/relationalai/experimental/paths/graph.py +185 -0
  254. v0/relationalai/experimental/paths/path_algorithms/find_paths.py +280 -0
  255. v0/relationalai/experimental/paths/path_algorithms/one_sided_ball_repetition.py +26 -0
  256. v0/relationalai/experimental/paths/path_algorithms/one_sided_ball_upto.py +111 -0
  257. v0/relationalai/experimental/paths/path_algorithms/single.py +59 -0
  258. v0/relationalai/experimental/paths/path_algorithms/two_sided_balls_repetition.py +39 -0
  259. v0/relationalai/experimental/paths/path_algorithms/two_sided_balls_upto.py +103 -0
  260. v0/relationalai/experimental/paths/path_algorithms/usp-old.py +130 -0
  261. v0/relationalai/experimental/paths/path_algorithms/usp-tuple.py +183 -0
  262. v0/relationalai/experimental/paths/path_algorithms/usp.py +150 -0
  263. v0/relationalai/experimental/paths/product_graph.py +93 -0
  264. v0/relationalai/experimental/paths/rpq/automaton.py +584 -0
  265. v0/relationalai/experimental/paths/rpq/diagnostics.py +56 -0
  266. v0/relationalai/experimental/paths/rpq/rpq.py +378 -0
  267. v0/relationalai/experimental/paths/tests/tests_limit_sp_max_length.py +90 -0
  268. v0/relationalai/experimental/paths/tests/tests_limit_sp_multiple.py +119 -0
  269. v0/relationalai/experimental/paths/tests/tests_limit_sp_single.py +104 -0
  270. v0/relationalai/experimental/paths/tests/tests_limit_walks_multiple.py +113 -0
  271. v0/relationalai/experimental/paths/tests/tests_limit_walks_single.py +149 -0
  272. v0/relationalai/experimental/paths/tests/tests_one_sided_ball_repetition_multiple.py +70 -0
  273. v0/relationalai/experimental/paths/tests/tests_one_sided_ball_repetition_single.py +64 -0
  274. v0/relationalai/experimental/paths/tests/tests_one_sided_ball_upto_multiple.py +115 -0
  275. v0/relationalai/experimental/paths/tests/tests_one_sided_ball_upto_single.py +75 -0
  276. v0/relationalai/experimental/paths/tests/tests_single_paths.py +152 -0
  277. v0/relationalai/experimental/paths/tests/tests_single_walks.py +208 -0
  278. v0/relationalai/experimental/paths/tests/tests_single_walks_undirected.py +297 -0
  279. v0/relationalai/experimental/paths/tests/tests_two_sided_balls_repetition_multiple.py +107 -0
  280. v0/relationalai/experimental/paths/tests/tests_two_sided_balls_repetition_single.py +76 -0
  281. v0/relationalai/experimental/paths/tests/tests_two_sided_balls_upto_multiple.py +76 -0
  282. v0/relationalai/experimental/paths/tests/tests_two_sided_balls_upto_single.py +110 -0
  283. v0/relationalai/experimental/paths/tests/tests_usp_nsp_multiple.py +229 -0
  284. v0/relationalai/experimental/paths/tests/tests_usp_nsp_single.py +108 -0
  285. v0/relationalai/experimental/paths/tree_agg.py +168 -0
  286. v0/relationalai/experimental/paths/utilities/iterators.py +27 -0
  287. v0/relationalai/experimental/paths/utilities/prefix_sum.py +91 -0
  288. v0/relationalai/experimental/solvers.py +1087 -0
  289. v0/relationalai/loaders/__init__.py +0 -0
  290. v0/relationalai/loaders/csv.py +195 -0
  291. v0/relationalai/loaders/loader.py +177 -0
  292. v0/relationalai/loaders/types.py +23 -0
  293. v0/relationalai/rel_emitter.py +373 -0
  294. v0/relationalai/rel_utils.py +185 -0
  295. v0/relationalai/semantics/__init__.py +29 -0
  296. v0/relationalai/semantics/devtools/benchmark_lqp.py +536 -0
  297. v0/relationalai/semantics/devtools/compilation_manager.py +294 -0
  298. v0/relationalai/semantics/devtools/extract_lqp.py +110 -0
  299. v0/relationalai/semantics/internal/internal.py +3785 -0
  300. v0/relationalai/semantics/internal/snowflake.py +324 -0
  301. v0/relationalai/semantics/lqp/builtins.py +16 -0
  302. v0/relationalai/semantics/lqp/compiler.py +22 -0
  303. v0/relationalai/semantics/lqp/constructors.py +68 -0
  304. v0/relationalai/semantics/lqp/executor.py +469 -0
  305. v0/relationalai/semantics/lqp/intrinsics.py +24 -0
  306. v0/relationalai/semantics/lqp/model2lqp.py +839 -0
  307. v0/relationalai/semantics/lqp/passes.py +680 -0
  308. v0/relationalai/semantics/lqp/primitives.py +252 -0
  309. v0/relationalai/semantics/lqp/result_helpers.py +202 -0
  310. v0/relationalai/semantics/lqp/rewrite/annotate_constraints.py +57 -0
  311. v0/relationalai/semantics/lqp/rewrite/cdc.py +216 -0
  312. v0/relationalai/semantics/lqp/rewrite/extract_common.py +338 -0
  313. v0/relationalai/semantics/lqp/rewrite/extract_keys.py +449 -0
  314. v0/relationalai/semantics/lqp/rewrite/function_annotations.py +114 -0
  315. v0/relationalai/semantics/lqp/rewrite/functional_dependencies.py +314 -0
  316. v0/relationalai/semantics/lqp/rewrite/quantify_vars.py +296 -0
  317. v0/relationalai/semantics/lqp/rewrite/splinter.py +76 -0
  318. v0/relationalai/semantics/lqp/types.py +101 -0
  319. v0/relationalai/semantics/lqp/utils.py +160 -0
  320. v0/relationalai/semantics/lqp/validators.py +57 -0
  321. v0/relationalai/semantics/metamodel/__init__.py +40 -0
  322. v0/relationalai/semantics/metamodel/builtins.py +774 -0
  323. v0/relationalai/semantics/metamodel/compiler.py +133 -0
  324. v0/relationalai/semantics/metamodel/dependency.py +862 -0
  325. v0/relationalai/semantics/metamodel/executor.py +61 -0
  326. v0/relationalai/semantics/metamodel/factory.py +287 -0
  327. v0/relationalai/semantics/metamodel/helpers.py +361 -0
  328. v0/relationalai/semantics/metamodel/rewrite/discharge_constraints.py +39 -0
  329. v0/relationalai/semantics/metamodel/rewrite/dnf_union_splitter.py +210 -0
  330. v0/relationalai/semantics/metamodel/rewrite/extract_nested_logicals.py +78 -0
  331. v0/relationalai/semantics/metamodel/rewrite/flatten.py +549 -0
  332. v0/relationalai/semantics/metamodel/rewrite/format_outputs.py +165 -0
  333. v0/relationalai/semantics/metamodel/typer/checker.py +353 -0
  334. v0/relationalai/semantics/metamodel/typer/typer.py +1395 -0
  335. v0/relationalai/semantics/reasoners/__init__.py +10 -0
  336. v0/relationalai/semantics/reasoners/graph/__init__.py +37 -0
  337. v0/relationalai/semantics/reasoners/graph/core.py +9020 -0
  338. v0/relationalai/semantics/reasoners/optimization/__init__.py +68 -0
  339. v0/relationalai/semantics/reasoners/optimization/common.py +88 -0
  340. v0/relationalai/semantics/reasoners/optimization/solvers_dev.py +568 -0
  341. v0/relationalai/semantics/reasoners/optimization/solvers_pb.py +1163 -0
  342. v0/relationalai/semantics/rel/builtins.py +40 -0
  343. v0/relationalai/semantics/rel/compiler.py +989 -0
  344. v0/relationalai/semantics/rel/executor.py +359 -0
  345. v0/relationalai/semantics/rel/rel.py +482 -0
  346. v0/relationalai/semantics/rel/rel_utils.py +276 -0
  347. v0/relationalai/semantics/snowflake/__init__.py +3 -0
  348. v0/relationalai/semantics/sql/compiler.py +2503 -0
  349. v0/relationalai/semantics/sql/executor/duck_db.py +52 -0
  350. v0/relationalai/semantics/sql/executor/result_helpers.py +64 -0
  351. v0/relationalai/semantics/sql/executor/snowflake.py +145 -0
  352. v0/relationalai/semantics/sql/rewrite/denormalize.py +222 -0
  353. v0/relationalai/semantics/sql/rewrite/double_negation.py +49 -0
  354. v0/relationalai/semantics/sql/rewrite/recursive_union.py +127 -0
  355. v0/relationalai/semantics/sql/rewrite/sort_output_query.py +246 -0
  356. v0/relationalai/semantics/sql/sql.py +504 -0
  357. v0/relationalai/semantics/std/__init__.py +54 -0
  358. v0/relationalai/semantics/std/constraints.py +43 -0
  359. v0/relationalai/semantics/std/datetime.py +363 -0
  360. v0/relationalai/semantics/std/decimals.py +62 -0
  361. v0/relationalai/semantics/std/floats.py +7 -0
  362. v0/relationalai/semantics/std/integers.py +22 -0
  363. v0/relationalai/semantics/std/math.py +141 -0
  364. v0/relationalai/semantics/std/pragmas.py +11 -0
  365. v0/relationalai/semantics/std/re.py +83 -0
  366. v0/relationalai/semantics/std/std.py +14 -0
  367. v0/relationalai/semantics/std/strings.py +63 -0
  368. v0/relationalai/semantics/tests/__init__.py +0 -0
  369. v0/relationalai/semantics/tests/test_snapshot_abstract.py +143 -0
  370. v0/relationalai/semantics/tests/test_snapshot_base.py +9 -0
  371. v0/relationalai/semantics/tests/utils.py +46 -0
  372. v0/relationalai/std/__init__.py +70 -0
  373. v0/relationalai/tools/__init__.py +0 -0
  374. v0/relationalai/tools/cli.py +1940 -0
  375. v0/relationalai/tools/cli_controls.py +1826 -0
  376. v0/relationalai/tools/cli_helpers.py +390 -0
  377. v0/relationalai/tools/debugger.py +183 -0
  378. v0/relationalai/tools/debugger_client.py +109 -0
  379. v0/relationalai/tools/debugger_server.py +302 -0
  380. v0/relationalai/tools/dev.py +685 -0
  381. v0/relationalai/tools/qb_debugger.py +425 -0
  382. v0/relationalai/util/clean_up_databases.py +95 -0
  383. v0/relationalai/util/format.py +123 -0
  384. v0/relationalai/util/list_databases.py +9 -0
  385. v0/relationalai/util/otel_configuration.py +25 -0
  386. v0/relationalai/util/otel_handler.py +484 -0
  387. v0/relationalai/util/snowflake_handler.py +88 -0
  388. v0/relationalai/util/span_format_test.py +43 -0
  389. v0/relationalai/util/span_tracker.py +207 -0
  390. v0/relationalai/util/spans_file_handler.py +72 -0
  391. v0/relationalai/util/tracing_handler.py +34 -0
  392. frontend/debugger/dist/.gitignore +0 -2
  393. frontend/debugger/dist/assets/favicon-Dy0ZgA6N.png +0 -0
  394. frontend/debugger/dist/assets/index-Cssla-O7.js +0 -208
  395. frontend/debugger/dist/assets/index-DlHsYx1V.css +0 -9
  396. frontend/debugger/dist/index.html +0 -17
  397. relationalai/clients/azure.py +0 -477
  398. relationalai/clients/client.py +0 -912
  399. relationalai/clients/config.py +0 -673
  400. relationalai/clients/direct_access_client.py +0 -118
  401. relationalai/clients/export_procedure.py.jinja +0 -249
  402. relationalai/clients/hash_util.py +0 -31
  403. relationalai/clients/local.py +0 -571
  404. relationalai/clients/profile_polling.py +0 -73
  405. relationalai/clients/result_helpers.py +0 -420
  406. relationalai/clients/snowflake.py +0 -3869
  407. relationalai/clients/types.py +0 -113
  408. relationalai/clients/use_index_poller.py +0 -980
  409. relationalai/clients/util.py +0 -356
  410. relationalai/debugging.py +0 -389
  411. relationalai/dsl.py +0 -1749
  412. relationalai/early_access/builder/__init__.py +0 -30
  413. relationalai/early_access/builder/builder/__init__.py +0 -35
  414. relationalai/early_access/builder/snowflake/__init__.py +0 -12
  415. relationalai/early_access/builder/std/__init__.py +0 -25
  416. relationalai/early_access/builder/std/decimals/__init__.py +0 -12
  417. relationalai/early_access/builder/std/integers/__init__.py +0 -12
  418. relationalai/early_access/builder/std/math/__init__.py +0 -12
  419. relationalai/early_access/builder/std/strings/__init__.py +0 -14
  420. relationalai/early_access/devtools/__init__.py +0 -12
  421. relationalai/early_access/devtools/benchmark_lqp/__init__.py +0 -12
  422. relationalai/early_access/devtools/extract_lqp/__init__.py +0 -12
  423. relationalai/early_access/dsl/adapters/orm/adapter_qb.py +0 -427
  424. relationalai/early_access/dsl/adapters/orm/parser.py +0 -636
  425. relationalai/early_access/dsl/adapters/owl/adapter.py +0 -176
  426. relationalai/early_access/dsl/adapters/owl/parser.py +0 -160
  427. relationalai/early_access/dsl/bindings/common.py +0 -402
  428. relationalai/early_access/dsl/bindings/csv.py +0 -170
  429. relationalai/early_access/dsl/bindings/legacy/binding_models.py +0 -143
  430. relationalai/early_access/dsl/bindings/snowflake.py +0 -64
  431. relationalai/early_access/dsl/codegen/binder.py +0 -411
  432. relationalai/early_access/dsl/codegen/common.py +0 -79
  433. relationalai/early_access/dsl/codegen/helpers.py +0 -23
  434. relationalai/early_access/dsl/codegen/relations.py +0 -700
  435. relationalai/early_access/dsl/codegen/weaver.py +0 -417
  436. relationalai/early_access/dsl/core/builders/__init__.py +0 -47
  437. relationalai/early_access/dsl/core/builders/logic.py +0 -19
  438. relationalai/early_access/dsl/core/builders/scalar_constraint.py +0 -11
  439. relationalai/early_access/dsl/core/constraints/predicate/atomic.py +0 -455
  440. relationalai/early_access/dsl/core/constraints/predicate/universal.py +0 -73
  441. relationalai/early_access/dsl/core/constraints/scalar.py +0 -310
  442. relationalai/early_access/dsl/core/context.py +0 -13
  443. relationalai/early_access/dsl/core/cset.py +0 -132
  444. relationalai/early_access/dsl/core/exprs/__init__.py +0 -116
  445. relationalai/early_access/dsl/core/exprs/relational.py +0 -18
  446. relationalai/early_access/dsl/core/exprs/scalar.py +0 -412
  447. relationalai/early_access/dsl/core/instances.py +0 -44
  448. relationalai/early_access/dsl/core/logic/__init__.py +0 -193
  449. relationalai/early_access/dsl/core/logic/aggregation.py +0 -98
  450. relationalai/early_access/dsl/core/logic/exists.py +0 -223
  451. relationalai/early_access/dsl/core/logic/helper.py +0 -163
  452. relationalai/early_access/dsl/core/namespaces.py +0 -32
  453. relationalai/early_access/dsl/core/relations.py +0 -276
  454. relationalai/early_access/dsl/core/rules.py +0 -112
  455. relationalai/early_access/dsl/core/std/__init__.py +0 -45
  456. relationalai/early_access/dsl/core/temporal/recall.py +0 -6
  457. relationalai/early_access/dsl/core/types/__init__.py +0 -270
  458. relationalai/early_access/dsl/core/types/concepts.py +0 -128
  459. relationalai/early_access/dsl/core/types/constrained/__init__.py +0 -267
  460. relationalai/early_access/dsl/core/types/constrained/nominal.py +0 -143
  461. relationalai/early_access/dsl/core/types/constrained/subtype.py +0 -124
  462. relationalai/early_access/dsl/core/types/standard.py +0 -92
  463. relationalai/early_access/dsl/core/types/unconstrained.py +0 -50
  464. relationalai/early_access/dsl/core/types/variables.py +0 -203
  465. relationalai/early_access/dsl/ir/compiler.py +0 -318
  466. relationalai/early_access/dsl/ir/executor.py +0 -260
  467. relationalai/early_access/dsl/ontologies/constraints.py +0 -88
  468. relationalai/early_access/dsl/ontologies/export.py +0 -30
  469. relationalai/early_access/dsl/ontologies/models.py +0 -453
  470. relationalai/early_access/dsl/ontologies/python_printer.py +0 -303
  471. relationalai/early_access/dsl/ontologies/readings.py +0 -60
  472. relationalai/early_access/dsl/ontologies/relationships.py +0 -322
  473. relationalai/early_access/dsl/ontologies/roles.py +0 -87
  474. relationalai/early_access/dsl/ontologies/subtyping.py +0 -55
  475. relationalai/early_access/dsl/orm/constraints.py +0 -438
  476. relationalai/early_access/dsl/orm/measures/dimensions.py +0 -200
  477. relationalai/early_access/dsl/orm/measures/initializer.py +0 -16
  478. relationalai/early_access/dsl/orm/measures/measure_rules.py +0 -275
  479. relationalai/early_access/dsl/orm/measures/measures.py +0 -299
  480. relationalai/early_access/dsl/orm/measures/role_exprs.py +0 -268
  481. relationalai/early_access/dsl/orm/models.py +0 -256
  482. relationalai/early_access/dsl/orm/object_oriented_printer.py +0 -344
  483. relationalai/early_access/dsl/orm/printer.py +0 -469
  484. relationalai/early_access/dsl/orm/reasoners.py +0 -480
  485. relationalai/early_access/dsl/orm/relations.py +0 -19
  486. relationalai/early_access/dsl/orm/relationships.py +0 -251
  487. relationalai/early_access/dsl/orm/types.py +0 -42
  488. relationalai/early_access/dsl/orm/utils.py +0 -79
  489. relationalai/early_access/dsl/orm/verb.py +0 -204
  490. relationalai/early_access/dsl/physical_metadata/tables.py +0 -133
  491. relationalai/early_access/dsl/relations.py +0 -170
  492. relationalai/early_access/dsl/rulesets.py +0 -69
  493. relationalai/early_access/dsl/schemas/__init__.py +0 -450
  494. relationalai/early_access/dsl/schemas/builder.py +0 -48
  495. relationalai/early_access/dsl/schemas/comp_names.py +0 -51
  496. relationalai/early_access/dsl/schemas/components.py +0 -203
  497. relationalai/early_access/dsl/schemas/contexts.py +0 -156
  498. relationalai/early_access/dsl/schemas/exprs.py +0 -89
  499. relationalai/early_access/dsl/schemas/fragments.py +0 -464
  500. relationalai/early_access/dsl/serialization.py +0 -79
  501. relationalai/early_access/dsl/serialize/exporter.py +0 -163
  502. relationalai/early_access/dsl/snow/api.py +0 -104
  503. relationalai/early_access/dsl/snow/common.py +0 -76
  504. relationalai/early_access/dsl/state_mgmt/__init__.py +0 -129
  505. relationalai/early_access/dsl/state_mgmt/state_charts.py +0 -125
  506. relationalai/early_access/dsl/state_mgmt/transitions.py +0 -130
  507. relationalai/early_access/dsl/types/__init__.py +0 -40
  508. relationalai/early_access/dsl/types/concepts.py +0 -12
  509. relationalai/early_access/dsl/types/entities.py +0 -135
  510. relationalai/early_access/dsl/types/values.py +0 -17
  511. relationalai/early_access/dsl/utils.py +0 -102
  512. relationalai/early_access/graphs/__init__.py +0 -13
  513. relationalai/early_access/lqp/__init__.py +0 -12
  514. relationalai/early_access/lqp/compiler/__init__.py +0 -12
  515. relationalai/early_access/lqp/constructors/__init__.py +0 -18
  516. relationalai/early_access/lqp/executor/__init__.py +0 -12
  517. relationalai/early_access/lqp/ir/__init__.py +0 -12
  518. relationalai/early_access/lqp/passes/__init__.py +0 -12
  519. relationalai/early_access/lqp/pragmas/__init__.py +0 -12
  520. relationalai/early_access/lqp/primitives/__init__.py +0 -12
  521. relationalai/early_access/lqp/types/__init__.py +0 -12
  522. relationalai/early_access/lqp/utils/__init__.py +0 -12
  523. relationalai/early_access/lqp/validators/__init__.py +0 -12
  524. relationalai/early_access/metamodel/__init__.py +0 -58
  525. relationalai/early_access/metamodel/builtins/__init__.py +0 -12
  526. relationalai/early_access/metamodel/compiler/__init__.py +0 -12
  527. relationalai/early_access/metamodel/dependency/__init__.py +0 -12
  528. relationalai/early_access/metamodel/factory/__init__.py +0 -17
  529. relationalai/early_access/metamodel/helpers/__init__.py +0 -12
  530. relationalai/early_access/metamodel/ir/__init__.py +0 -14
  531. relationalai/early_access/metamodel/rewrite/__init__.py +0 -7
  532. relationalai/early_access/metamodel/typer/__init__.py +0 -3
  533. relationalai/early_access/metamodel/typer/typer/__init__.py +0 -12
  534. relationalai/early_access/metamodel/types/__init__.py +0 -15
  535. relationalai/early_access/metamodel/util/__init__.py +0 -15
  536. relationalai/early_access/metamodel/visitor/__init__.py +0 -12
  537. relationalai/early_access/rel/__init__.py +0 -12
  538. relationalai/early_access/rel/executor/__init__.py +0 -12
  539. relationalai/early_access/rel/rel_utils/__init__.py +0 -12
  540. relationalai/early_access/rel/rewrite/__init__.py +0 -7
  541. relationalai/early_access/solvers/__init__.py +0 -19
  542. relationalai/early_access/sql/__init__.py +0 -11
  543. relationalai/early_access/sql/executor/__init__.py +0 -3
  544. relationalai/early_access/sql/rewrite/__init__.py +0 -3
  545. relationalai/early_access/tests/logging/__init__.py +0 -12
  546. relationalai/early_access/tests/test_snapshot_base/__init__.py +0 -12
  547. relationalai/early_access/tests/utils/__init__.py +0 -12
  548. relationalai/environments/__init__.py +0 -35
  549. relationalai/environments/base.py +0 -381
  550. relationalai/environments/colab.py +0 -14
  551. relationalai/environments/generic.py +0 -71
  552. relationalai/environments/ipython.py +0 -68
  553. relationalai/environments/jupyter.py +0 -9
  554. relationalai/environments/snowbook.py +0 -169
  555. relationalai/errors.py +0 -2455
  556. relationalai/experimental/SF.py +0 -38
  557. relationalai/experimental/inspect.py +0 -47
  558. relationalai/experimental/pathfinder/__init__.py +0 -158
  559. relationalai/experimental/pathfinder/api.py +0 -160
  560. relationalai/experimental/pathfinder/automaton.py +0 -584
  561. relationalai/experimental/pathfinder/bridge.py +0 -226
  562. relationalai/experimental/pathfinder/compiler.py +0 -416
  563. relationalai/experimental/pathfinder/datalog.py +0 -214
  564. relationalai/experimental/pathfinder/diagnostics.py +0 -56
  565. relationalai/experimental/pathfinder/filter.py +0 -236
  566. relationalai/experimental/pathfinder/glushkov.py +0 -439
  567. relationalai/experimental/pathfinder/options.py +0 -265
  568. relationalai/experimental/pathfinder/pathfinder-v0.7.0.rel +0 -1951
  569. relationalai/experimental/pathfinder/rpq.py +0 -344
  570. relationalai/experimental/pathfinder/transition.py +0 -200
  571. relationalai/experimental/pathfinder/utils.py +0 -26
  572. relationalai/experimental/paths/README.md +0 -107
  573. relationalai/experimental/paths/api.py +0 -143
  574. relationalai/experimental/paths/benchmarks/grid_graph.py +0 -37
  575. relationalai/experimental/paths/code_organization.md +0 -2
  576. relationalai/experimental/paths/examples/Movies.ipynb +0 -16328
  577. relationalai/experimental/paths/examples/basic_example.py +0 -40
  578. relationalai/experimental/paths/examples/minimal_engine_warmup.py +0 -3
  579. relationalai/experimental/paths/examples/movie_example.py +0 -77
  580. relationalai/experimental/paths/examples/movies_data/actedin.csv +0 -193
  581. relationalai/experimental/paths/examples/movies_data/directed.csv +0 -45
  582. relationalai/experimental/paths/examples/movies_data/follows.csv +0 -7
  583. relationalai/experimental/paths/examples/movies_data/movies.csv +0 -39
  584. relationalai/experimental/paths/examples/movies_data/person.csv +0 -134
  585. relationalai/experimental/paths/examples/movies_data/produced.csv +0 -16
  586. relationalai/experimental/paths/examples/movies_data/ratings.csv +0 -10
  587. relationalai/experimental/paths/examples/movies_data/wrote.csv +0 -11
  588. relationalai/experimental/paths/examples/paths_benchmark.py +0 -115
  589. relationalai/experimental/paths/examples/paths_example.py +0 -116
  590. relationalai/experimental/paths/examples/pattern_to_automaton.py +0 -28
  591. relationalai/experimental/paths/find_paths_via_automaton.py +0 -85
  592. relationalai/experimental/paths/graph.py +0 -185
  593. relationalai/experimental/paths/path_algorithms/find_paths.py +0 -280
  594. relationalai/experimental/paths/path_algorithms/one_sided_ball_repetition.py +0 -26
  595. relationalai/experimental/paths/path_algorithms/one_sided_ball_upto.py +0 -111
  596. relationalai/experimental/paths/path_algorithms/single.py +0 -59
  597. relationalai/experimental/paths/path_algorithms/two_sided_balls_repetition.py +0 -39
  598. relationalai/experimental/paths/path_algorithms/two_sided_balls_upto.py +0 -103
  599. relationalai/experimental/paths/path_algorithms/usp-old.py +0 -130
  600. relationalai/experimental/paths/path_algorithms/usp-tuple.py +0 -183
  601. relationalai/experimental/paths/path_algorithms/usp.py +0 -150
  602. relationalai/experimental/paths/product_graph.py +0 -93
  603. relationalai/experimental/paths/rpq/automaton.py +0 -584
  604. relationalai/experimental/paths/rpq/diagnostics.py +0 -56
  605. relationalai/experimental/paths/rpq/rpq.py +0 -378
  606. relationalai/experimental/paths/tests/tests_limit_sp_max_length.py +0 -90
  607. relationalai/experimental/paths/tests/tests_limit_sp_multiple.py +0 -119
  608. relationalai/experimental/paths/tests/tests_limit_sp_single.py +0 -104
  609. relationalai/experimental/paths/tests/tests_limit_walks_multiple.py +0 -113
  610. relationalai/experimental/paths/tests/tests_limit_walks_single.py +0 -149
  611. relationalai/experimental/paths/tests/tests_one_sided_ball_repetition_multiple.py +0 -70
  612. relationalai/experimental/paths/tests/tests_one_sided_ball_repetition_single.py +0 -64
  613. relationalai/experimental/paths/tests/tests_one_sided_ball_upto_multiple.py +0 -115
  614. relationalai/experimental/paths/tests/tests_one_sided_ball_upto_single.py +0 -75
  615. relationalai/experimental/paths/tests/tests_single_paths.py +0 -152
  616. relationalai/experimental/paths/tests/tests_single_walks.py +0 -208
  617. relationalai/experimental/paths/tests/tests_single_walks_undirected.py +0 -297
  618. relationalai/experimental/paths/tests/tests_two_sided_balls_repetition_multiple.py +0 -107
  619. relationalai/experimental/paths/tests/tests_two_sided_balls_repetition_single.py +0 -76
  620. relationalai/experimental/paths/tests/tests_two_sided_balls_upto_multiple.py +0 -76
  621. relationalai/experimental/paths/tests/tests_two_sided_balls_upto_single.py +0 -110
  622. relationalai/experimental/paths/tests/tests_usp_nsp_multiple.py +0 -229
  623. relationalai/experimental/paths/tests/tests_usp_nsp_single.py +0 -108
  624. relationalai/experimental/paths/tree_agg.py +0 -168
  625. relationalai/experimental/paths/utilities/iterators.py +0 -27
  626. relationalai/experimental/paths/utilities/prefix_sum.py +0 -91
  627. relationalai/experimental/solvers.py +0 -1087
  628. relationalai/loaders/csv.py +0 -195
  629. relationalai/loaders/loader.py +0 -177
  630. relationalai/loaders/types.py +0 -23
  631. relationalai/rel_emitter.py +0 -373
  632. relationalai/rel_utils.py +0 -185
  633. relationalai/semantics/designs/query_builder/identify_by.md +0 -106
  634. relationalai/semantics/devtools/benchmark_lqp.py +0 -536
  635. relationalai/semantics/devtools/compilation_manager.py +0 -294
  636. relationalai/semantics/devtools/extract_lqp.py +0 -110
  637. relationalai/semantics/internal/internal.py +0 -3785
  638. relationalai/semantics/internal/snowflake.py +0 -324
  639. relationalai/semantics/lqp/README.md +0 -34
  640. relationalai/semantics/lqp/builtins.py +0 -16
  641. relationalai/semantics/lqp/compiler.py +0 -22
  642. relationalai/semantics/lqp/constructors.py +0 -68
  643. relationalai/semantics/lqp/executor.py +0 -469
  644. relationalai/semantics/lqp/intrinsics.py +0 -24
  645. relationalai/semantics/lqp/model2lqp.py +0 -839
  646. relationalai/semantics/lqp/passes.py +0 -680
  647. relationalai/semantics/lqp/primitives.py +0 -252
  648. relationalai/semantics/lqp/result_helpers.py +0 -202
  649. relationalai/semantics/lqp/rewrite/annotate_constraints.py +0 -57
  650. relationalai/semantics/lqp/rewrite/cdc.py +0 -216
  651. relationalai/semantics/lqp/rewrite/extract_common.py +0 -338
  652. relationalai/semantics/lqp/rewrite/extract_keys.py +0 -449
  653. relationalai/semantics/lqp/rewrite/function_annotations.py +0 -114
  654. relationalai/semantics/lqp/rewrite/functional_dependencies.py +0 -314
  655. relationalai/semantics/lqp/rewrite/quantify_vars.py +0 -296
  656. relationalai/semantics/lqp/rewrite/splinter.py +0 -76
  657. relationalai/semantics/lqp/types.py +0 -101
  658. relationalai/semantics/lqp/utils.py +0 -160
  659. relationalai/semantics/lqp/validators.py +0 -57
  660. relationalai/semantics/metamodel/compiler.py +0 -133
  661. relationalai/semantics/metamodel/dependency.py +0 -862
  662. relationalai/semantics/metamodel/executor.py +0 -61
  663. relationalai/semantics/metamodel/factory.py +0 -287
  664. relationalai/semantics/metamodel/helpers.py +0 -361
  665. relationalai/semantics/metamodel/rewrite/discharge_constraints.py +0 -39
  666. relationalai/semantics/metamodel/rewrite/dnf_union_splitter.py +0 -210
  667. relationalai/semantics/metamodel/rewrite/extract_nested_logicals.py +0 -78
  668. relationalai/semantics/metamodel/rewrite/flatten.py +0 -549
  669. relationalai/semantics/metamodel/rewrite/format_outputs.py +0 -165
  670. relationalai/semantics/metamodel/typer/checker.py +0 -353
  671. relationalai/semantics/metamodel/typer/typer.py +0 -1395
  672. relationalai/semantics/reasoners/__init__.py +0 -10
  673. relationalai/semantics/reasoners/graph/README.md +0 -620
  674. relationalai/semantics/reasoners/graph/__init__.py +0 -37
  675. relationalai/semantics/reasoners/graph/core.py +0 -9020
  676. relationalai/semantics/reasoners/graph/design/beyond_demand_transform.md +0 -797
  677. relationalai/semantics/reasoners/graph/tests/README.md +0 -21
  678. relationalai/semantics/reasoners/optimization/__init__.py +0 -68
  679. relationalai/semantics/reasoners/optimization/common.py +0 -88
  680. relationalai/semantics/reasoners/optimization/solvers_dev.py +0 -568
  681. relationalai/semantics/reasoners/optimization/solvers_pb.py +0 -1163
  682. relationalai/semantics/rel/builtins.py +0 -40
  683. relationalai/semantics/rel/compiler.py +0 -989
  684. relationalai/semantics/rel/executor.py +0 -359
  685. relationalai/semantics/rel/rel.py +0 -482
  686. relationalai/semantics/rel/rel_utils.py +0 -276
  687. relationalai/semantics/snowflake/__init__.py +0 -3
  688. relationalai/semantics/sql/compiler.py +0 -2503
  689. relationalai/semantics/sql/executor/duck_db.py +0 -52
  690. relationalai/semantics/sql/executor/result_helpers.py +0 -64
  691. relationalai/semantics/sql/executor/snowflake.py +0 -145
  692. relationalai/semantics/sql/rewrite/denormalize.py +0 -222
  693. relationalai/semantics/sql/rewrite/double_negation.py +0 -49
  694. relationalai/semantics/sql/rewrite/recursive_union.py +0 -127
  695. relationalai/semantics/sql/rewrite/sort_output_query.py +0 -246
  696. relationalai/semantics/sql/sql.py +0 -504
  697. relationalai/semantics/std/pragmas.py +0 -11
  698. relationalai/semantics/std/std.py +0 -14
  699. relationalai/semantics/tests/test_snapshot_abstract.py +0 -143
  700. relationalai/semantics/tests/test_snapshot_base.py +0 -9
  701. relationalai/semantics/tests/utils.py +0 -46
  702. relationalai/std/__init__.py +0 -70
  703. relationalai/tools/cli.py +0 -1940
  704. relationalai/tools/cli_controls.py +0 -1826
  705. relationalai/tools/cli_helpers.py +0 -390
  706. relationalai/tools/debugger_client.py +0 -109
  707. relationalai/tools/debugger_server.py +0 -302
  708. relationalai/tools/dev.py +0 -685
  709. relationalai/tools/notes +0 -7
  710. relationalai/tools/qb_debugger.py +0 -425
  711. relationalai/util/clean_up_databases.py +0 -95
  712. relationalai/util/list_databases.py +0 -9
  713. relationalai/util/otel_configuration.py +0 -25
  714. relationalai/util/otel_handler.py +0 -484
  715. relationalai/util/snowflake_handler.py +0 -88
  716. relationalai/util/span_format_test.py +0 -43
  717. relationalai/util/span_tracker.py +0 -207
  718. relationalai/util/spans_file_handler.py +0 -72
  719. relationalai/util/tracing_handler.py +0 -34
  720. relationalai-0.12.13.dist-info/METADATA +0 -74
  721. relationalai-0.12.13.dist-info/RECORD +0 -449
  722. relationalai-0.12.13.dist-info/WHEEL +0 -4
  723. relationalai-0.12.13.dist-info/entry_points.txt +0 -3
  724. relationalai-0.12.13.dist-info/licenses/LICENSE +0 -202
  725. relationalai_test_util/__init__.py +0 -4
  726. relationalai_test_util/fixtures.py +0 -228
  727. relationalai_test_util/snapshot.py +0 -252
  728. relationalai_test_util/traceback.py +0 -118
  729. /relationalai/{analysis → semantics/frontend}/__init__.py +0 -0
  730. /relationalai/{auth/__init__.py → semantics/metamodel/metamodel_compiler.py} +0 -0
  731. /relationalai/{early_access → shims}/__init__.py +0 -0
  732. {relationalai/early_access/dsl/adapters → v0/relationalai/analysis}/__init__.py +0 -0
  733. {relationalai → v0/relationalai}/analysis/mechanistic.py +0 -0
  734. {relationalai → v0/relationalai}/analysis/whynot.py +0 -0
  735. {relationalai/early_access/dsl/adapters/orm → v0/relationalai/auth}/__init__.py +0 -0
  736. {relationalai → v0/relationalai}/auth/jwt_generator.py +0 -0
  737. {relationalai → v0/relationalai}/auth/oauth_callback_server.py +0 -0
  738. {relationalai → v0/relationalai}/auth/token_handler.py +0 -0
  739. {relationalai → v0/relationalai}/auth/util.py +0 -0
  740. {relationalai → v0/relationalai}/clients/__init__.py +0 -0
  741. {relationalai → v0/relationalai}/clients/cache_store.py +0 -0
  742. {relationalai → v0/relationalai}/compiler.py +0 -0
  743. {relationalai → v0/relationalai}/dependencies.py +0 -0
  744. {relationalai → v0/relationalai}/docutils.py +0 -0
  745. {relationalai/early_access/dsl/adapters/owl → v0/relationalai/early_access}/__init__.py +0 -0
  746. {relationalai → v0/relationalai}/early_access/dsl/__init__.py +0 -0
  747. {relationalai/early_access/dsl/bindings → v0/relationalai/early_access/dsl/adapters}/__init__.py +0 -0
  748. {relationalai/early_access/dsl/bindings/legacy → v0/relationalai/early_access/dsl/adapters/orm}/__init__.py +0 -0
  749. {relationalai → v0/relationalai}/early_access/dsl/adapters/orm/model.py +0 -0
  750. {relationalai/early_access/dsl/codegen → v0/relationalai/early_access/dsl/adapters/owl}/__init__.py +0 -0
  751. {relationalai → v0/relationalai}/early_access/dsl/adapters/owl/model.py +0 -0
  752. {relationalai/early_access/dsl/core/temporal → v0/relationalai/early_access/dsl/bindings}/__init__.py +0 -0
  753. {relationalai/early_access/dsl/ir → v0/relationalai/early_access/dsl/bindings/legacy}/__init__.py +0 -0
  754. {relationalai/early_access/dsl/ontologies → v0/relationalai/early_access/dsl/codegen}/__init__.py +0 -0
  755. {relationalai → v0/relationalai}/early_access/dsl/constants.py +0 -0
  756. {relationalai → v0/relationalai}/early_access/dsl/core/__init__.py +0 -0
  757. {relationalai → v0/relationalai}/early_access/dsl/core/constraints/__init__.py +0 -0
  758. {relationalai → v0/relationalai}/early_access/dsl/core/constraints/predicate/__init__.py +0 -0
  759. {relationalai → v0/relationalai}/early_access/dsl/core/stack.py +0 -0
  760. {relationalai/early_access/dsl/orm → v0/relationalai/early_access/dsl/core/temporal}/__init__.py +0 -0
  761. {relationalai → v0/relationalai}/early_access/dsl/core/utils.py +0 -0
  762. {relationalai/early_access/dsl/orm/measures → v0/relationalai/early_access/dsl/ir}/__init__.py +0 -0
  763. {relationalai/early_access/dsl/physical_metadata → v0/relationalai/early_access/dsl/ontologies}/__init__.py +0 -0
  764. {relationalai → v0/relationalai}/early_access/dsl/ontologies/raw_source.py +0 -0
  765. {relationalai/early_access/dsl/serialize → v0/relationalai/early_access/dsl/orm}/__init__.py +0 -0
  766. {relationalai/early_access/dsl/snow → v0/relationalai/early_access/dsl/orm/measures}/__init__.py +0 -0
  767. {relationalai → v0/relationalai}/early_access/dsl/orm/reasoner_errors.py +0 -0
  768. {relationalai/loaders → v0/relationalai/early_access/dsl/physical_metadata}/__init__.py +0 -0
  769. {relationalai/semantics/tests → v0/relationalai/early_access/dsl/serialize}/__init__.py +0 -0
  770. {relationalai → v0/relationalai}/early_access/dsl/serialize/binding_model.py +0 -0
  771. {relationalai → v0/relationalai}/early_access/dsl/serialize/model.py +0 -0
  772. {relationalai/tools → v0/relationalai/early_access/dsl/snow}/__init__.py +0 -0
  773. {relationalai → v0/relationalai}/early_access/tests/__init__.py +0 -0
  774. {relationalai → v0/relationalai}/environments/ci.py +0 -0
  775. {relationalai → v0/relationalai}/environments/hex.py +0 -0
  776. {relationalai → v0/relationalai}/environments/terminal.py +0 -0
  777. {relationalai → v0/relationalai}/experimental/__init__.py +0 -0
  778. {relationalai → v0/relationalai}/experimental/graphs.py +0 -0
  779. {relationalai → v0/relationalai}/experimental/paths/__init__.py +0 -0
  780. {relationalai → v0/relationalai}/experimental/paths/benchmarks/__init__.py +0 -0
  781. {relationalai → v0/relationalai}/experimental/paths/path_algorithms/__init__.py +0 -0
  782. {relationalai → v0/relationalai}/experimental/paths/rpq/__init__.py +0 -0
  783. {relationalai → v0/relationalai}/experimental/paths/rpq/filter.py +0 -0
  784. {relationalai → v0/relationalai}/experimental/paths/rpq/glushkov.py +0 -0
  785. {relationalai → v0/relationalai}/experimental/paths/rpq/transition.py +0 -0
  786. {relationalai → v0/relationalai}/experimental/paths/utilities/__init__.py +0 -0
  787. {relationalai → v0/relationalai}/experimental/paths/utilities/utilities.py +0 -0
  788. {relationalai → v0/relationalai}/metagen.py +0 -0
  789. {relationalai → v0/relationalai}/metamodel.py +0 -0
  790. {relationalai → v0/relationalai}/rel.py +0 -0
  791. {relationalai → v0/relationalai}/semantics/devtools/__init__.py +0 -0
  792. {relationalai → v0/relationalai}/semantics/internal/__init__.py +0 -0
  793. {relationalai → v0/relationalai}/semantics/internal/annotations.py +0 -0
  794. {relationalai → v0/relationalai}/semantics/lqp/__init__.py +0 -0
  795. {relationalai → v0/relationalai}/semantics/lqp/ir.py +0 -0
  796. {relationalai → v0/relationalai}/semantics/lqp/pragmas.py +0 -0
  797. {relationalai → v0/relationalai}/semantics/lqp/rewrite/__init__.py +0 -0
  798. {relationalai → v0/relationalai}/semantics/metamodel/dataflow.py +0 -0
  799. {relationalai → v0/relationalai}/semantics/metamodel/ir.py +0 -0
  800. {relationalai → v0/relationalai}/semantics/metamodel/rewrite/__init__.py +0 -0
  801. {relationalai → v0/relationalai}/semantics/metamodel/typer/__init__.py +0 -0
  802. {relationalai → v0/relationalai}/semantics/metamodel/types.py +0 -0
  803. {relationalai → v0/relationalai}/semantics/metamodel/util.py +0 -0
  804. {relationalai → v0/relationalai}/semantics/metamodel/visitor.py +0 -0
  805. {relationalai → v0/relationalai}/semantics/reasoners/experimental/__init__.py +0 -0
  806. {relationalai → v0/relationalai}/semantics/rel/__init__.py +0 -0
  807. {relationalai → v0/relationalai}/semantics/sql/__init__.py +0 -0
  808. {relationalai → v0/relationalai}/semantics/sql/executor/__init__.py +0 -0
  809. {relationalai → v0/relationalai}/semantics/sql/rewrite/__init__.py +0 -0
  810. {relationalai → v0/relationalai}/semantics/tests/logging.py +0 -0
  811. {relationalai → v0/relationalai}/std/aggregates.py +0 -0
  812. {relationalai → v0/relationalai}/std/dates.py +0 -0
  813. {relationalai → v0/relationalai}/std/graphs.py +0 -0
  814. {relationalai → v0/relationalai}/std/inspect.py +0 -0
  815. {relationalai → v0/relationalai}/std/math.py +0 -0
  816. {relationalai → v0/relationalai}/std/re.py +0 -0
  817. {relationalai → v0/relationalai}/std/strings.py +0 -0
  818. {relationalai → v0/relationalai}/tools/cleanup_snapshots.py +0 -0
  819. {relationalai → v0/relationalai}/tools/constants.py +0 -0
  820. {relationalai → v0/relationalai}/tools/query_utils.py +0 -0
  821. {relationalai → v0/relationalai}/tools/snapshot_viewer.py +0 -0
  822. {relationalai → v0/relationalai}/util/__init__.py +0 -0
  823. {relationalai → v0/relationalai}/util/constants.py +0 -0
  824. {relationalai → v0/relationalai}/util/graph.py +0 -0
  825. {relationalai → v0/relationalai}/util/timeout.py +0 -0
@@ -0,0 +1,1087 @@
1
+ from __future__ import annotations
2
+ import time
3
+ from typing import Any, List, Optional
4
+ from dataclasses import dataclass
5
+ import textwrap
6
+ from .. import dsl, std
7
+ from ..std import rel
8
+ from ..metamodel import Builtins
9
+ from ..tools.cli_controls import Spinner
10
+ from ..tools.constants import DEFAULT_QUERY_TIMEOUT_MINS
11
+ from .. import debugging
12
+ from .. errors import ResponseStatusException
13
+ import uuid
14
+ import v0.relationalai
15
+ import json
16
+ from ..clients.util import poll_with_specified_overhead
17
+ from ..clients.snowflake import Resources as SnowflakeResources
18
+ from ..clients.snowflake import DirectAccessClient, DirectAccessResources
19
+ from ..util.timeout import calc_remaining_timeout_minutes
20
+
21
+ rel_sv = rel._tagged(Builtins.SingleValued)
22
+
23
+ APP_NAME = v0.relationalai.clients.snowflake.APP_NAME
24
+
25
+ ENGINE_TYPE_SOLVER = "SOLVER"
26
+ # TODO (dba) The ERP still uses `worker` instead of `engine`. Change
27
+ # this once we fix this in the ERP.
28
+ WORKER_ERRORS = ["worker is suspended", "create/resume", "worker not found", "no workers found", "worker was deleted"]
29
+ ENGINE_ERRORS = ["engine is suspended", "create/resume", "engine not found", "no engines found", "engine was deleted"]
30
+ ENGINE_NOT_READY_MSGS = ["worker is in pending", "worker is provisioning", "worker is not ready to accept jobs"]
31
+
32
+ # --------------------------------------------------
33
+ # SolverModel object.
34
+ # --------------------------------------------------
35
+
36
+ class SolverModel:
37
+ def __init__(self, graph:dsl.Graph):
38
+ self.graph = graph
39
+ self.id = dsl.next_id()
40
+ self.scope = f"solvermodel{self.id}_"
41
+ scope = self.scope
42
+ self.Variable = dsl.Type(graph, "variables", scope=scope)
43
+ self.MinObjective = dsl.Type(graph, "min_objectives", scope=scope)
44
+ self.MaxObjective = dsl.Type(graph, "max_objectives", scope=scope)
45
+ self.Constraint = dsl.Type(graph, "constraints", scope=scope)
46
+ self.Solution = dsl.Type(graph, "solutions", scope=scope)
47
+ self.components = [
48
+ (self.MinObjective, "minimization objectives"),
49
+ (self.MaxObjective, "maximization objectives"),
50
+ (self.Constraint, "constraints"),
51
+ ]
52
+ self.solve_index = 0
53
+ self.is_solved = False
54
+
55
+ # Install model helpers.
56
+ self.graph.install_raw(textwrap.dedent(f"""
57
+ @inline
58
+ def _solverlib_ho_appl(op, {{R}}, s): rel_primitive_solverlib_ho_appl(R, op, s)
59
+
60
+ @inline
61
+ def _solver_unwrap({{R}}, h, x...): exists((v) | R(v, x...) and pyrel_unwrap(v, h))
62
+
63
+ declare {scope}variable_name
64
+ declare {scope}component_name
65
+ declare {scope}serialized
66
+ declare {scope}primal_start
67
+
68
+ def {scope}component_string(h, s):
69
+ rel_primitive_solverlib_print_expr(
70
+ {scope}serialized[h], _solver_unwrap[{scope}variable_name], s
71
+ )
72
+
73
+ declare {scope}solve_output
74
+ """))
75
+ return None
76
+
77
+ # Add an entity to the variable set, (optionally) set a string name from the
78
+ # arguments and add domain constraints on the variable.
79
+ def variable(
80
+ self,
81
+ var, # variable entity
82
+ name_args:List|None=None, # list of strings to concatenate into a string name
83
+ type:str|None=None, # variable type: "integer" or "zero_one"
84
+ lower:int|float|None=None, # lower bound
85
+ upper:int|float|None=None, # upper bound
86
+ fixed:int|float|None=None, # fixed value
87
+ start:int|float|None=None, # (primal) start value
88
+ ):
89
+ if type not in {"integer", "zero_one", None}:
90
+ raise Exception(f"Invalid domain type: {type}.")
91
+ var.set(self.Variable)
92
+
93
+ # Set variable name.
94
+ if name_args:
95
+ var.set(**{f"{self.scope}variable_name": make_string(name_args)})
96
+
97
+ # Add domain constraints.
98
+ cons = []
99
+ if fixed is not None:
100
+ cons.append(eq(var, fixed))
101
+ if type == "zero_one":
102
+ cons.append(zero_one(var))
103
+ if lower is not None and upper is not None:
104
+ if type == "integer":
105
+ cons.append(integer_interval(var, lower, upper))
106
+ else:
107
+ cons.append(interval(var, lower, upper))
108
+ else:
109
+ if type == "integer":
110
+ cons.append(integer(var))
111
+ if lower is not None:
112
+ cons.append(gte(var, lower))
113
+ if upper is not None:
114
+ cons.append(lte(var, upper))
115
+ if len(cons) == 1:
116
+ self.constraint(cons[0])
117
+ elif len(cons) > 1:
118
+ self.constraint(and_(*cons))
119
+
120
+ # Set primal start.
121
+ if start is not None:
122
+ var.set(**{f"{self.scope}primal_start": start})
123
+ return var
124
+
125
+ # Get variable string name.
126
+ def variable_name(self, var):
127
+ return std.alias(getattr(var, f"{self.scope}variable_name"), "name")
128
+
129
+ # Add a constraint, minimization objective, or maximization objective.
130
+ def constraint(self, expr, name_args:List|None=None):
131
+ return self._add_component(self.Constraint, expr, name_args)
132
+
133
+ def min_objective(self, expr, name_args:List|None=None):
134
+ return self._add_component(self.MinObjective, expr, name_args)
135
+
136
+ def max_objective(self, expr, name_args:List|None=None):
137
+ return self._add_component(self.MaxObjective, expr, name_args)
138
+
139
+ def _add_component(self, typ, expr, name_args:List|None):
140
+ comp = typ.add(serialized=_wrap_expr(expr))
141
+ if name_args:
142
+ comp.set(component_name=make_string(name_args))
143
+ return comp
144
+
145
+ # Get component string name.
146
+ def component_name(self, comp):
147
+ return std.alias(comp.component_name, "name")
148
+
149
+ # Get serialized component string in human-readable format.
150
+ def component_string(self, comp):
151
+ return std.alias(comp.component_string, "string")
152
+
153
+ # Summarize the model by printing the number of variables and components.
154
+ # Use outside a rule/query.
155
+ def summarize(self):
156
+ with self.graph.query() as select:
157
+ vars = select(std.aggregates.count(self.Variable()))
158
+ s = f"Model has: {vars.results.iat[0, 0]} variables"
159
+ for (c_type, c_name) in self.components:
160
+ with self.graph.query() as select:
161
+ exprs = select(std.aggregates.count(c_type()))
162
+ if not exprs.results.empty:
163
+ s += f", {exprs.results.iat[0, 0]} {c_name}"
164
+ print(s)
165
+ return None
166
+
167
+ # Print the model in human-readable format. Use outside a rule/query.
168
+ def print(self):
169
+ with self.graph.query() as select:
170
+ vars = select(rel.last(getattr(rel, f"{self.scope}variable_name")))
171
+ print("variables:")
172
+ print(vars.results.to_string(index=False, header=False))
173
+ for (c_type, c_name) in self.components:
174
+ with self.graph.query() as select:
175
+ exprs = select(self.component_string(c_type()))
176
+ if not exprs.results.empty:
177
+ print(c_name + ":")
178
+ print(exprs.results.to_string(index=False, header=False))
179
+ return None
180
+
181
+ # Solve the model given a solver and solver options. Use outside a rule/query.
182
+ def solve(self, solver: Solver, log_to_console=True, **kwargs):
183
+ self.is_solved = False
184
+ self.solve_index += 1
185
+ self.solve_output = dsl.RelationNS([f"{self.scope}solve_output"], f"i_{self.solve_index}")
186
+
187
+ options = kwargs
188
+ options["version"] = 1
189
+
190
+ # Validate options.
191
+ for k, v in options.items():
192
+ if not isinstance(k, str):
193
+ raise Exception(f"Invalid parameter key. Expected string, got {type(k)} for {k}.")
194
+ if not isinstance(v, (int, float, str, bool)):
195
+ raise Exception(
196
+ f"Invalid parameter value. Expected string, integer, float, or boolean, got {type(v)} for {k}."
197
+ )
198
+
199
+ # Run the solve query and insert the solve_output result.
200
+ scope = self.scope
201
+ variable_name_string = f"{scope}variable_name" if "print_format" in options else "{}"
202
+ component_name_string = f"{scope}component_name" if "print_format" in options else "{}"
203
+
204
+ input_id = uuid.uuid4()
205
+ model_uri = f"snowflake://APP_STATE.RAI_INTERNAL_STAGE/job-inputs/solver/{input_id}/model.binpb"
206
+ sf_input_uri = f"snowflake://job-inputs/solver/{input_id}/model.binpb"
207
+
208
+ payload: dict[str, Any] = {"solver": solver.solver_name.lower()}
209
+ payload["options"] = options
210
+ payload["model_uri"] = sf_input_uri
211
+
212
+ rai_config = self.graph._config
213
+ query_timeout_mins = kwargs.get("query_timeout_mins", None)
214
+ if query_timeout_mins is None and (timeout_value := rai_config.get("query_timeout_mins", DEFAULT_QUERY_TIMEOUT_MINS)) is not None:
215
+ query_timeout_mins = int(timeout_value)
216
+ config_file_path = getattr(rai_config, 'file_path', None)
217
+ start_time = time.monotonic()
218
+ remaining_timeout_minutes = query_timeout_mins
219
+ response = self.graph.exec_raw(
220
+ textwrap.dedent(f"""
221
+ @inline
222
+ def {scope}specialized_components(t, h, s):
223
+ exists((v) | {{
224
+ (:min_objective, {scope}min_objectives);
225
+ (:max_objective, {scope}max_objectives);
226
+ (:constraint, {scope}constraints);
227
+ }}(t, v) and pyrel_unwrap(v, h) and {scope}serialized(v, s)
228
+ )
229
+
230
+ @no_diagnostics(:EXPERIMENTAL)
231
+ def {scope}model_string {{
232
+ rel_primitive_solverlib_model_string[{{
233
+ (:variable, _solver_unwrap[{scope}variables]);
234
+ {scope}specialized_components;
235
+ (:variable_name, _solver_unwrap[{variable_name_string}]);
236
+ (:expression_name, _solver_unwrap[{component_name_string}]);
237
+ (:primal_start, _solver_unwrap[{scope}primal_start]);
238
+ }}]
239
+ }}
240
+
241
+ ic model_not_empty("Solver model is empty.") requires not empty({scope}model_string)
242
+
243
+ def config[:envelope, :content_type]: "application/octet-stream"
244
+ def config[:envelope, :payload, :data]: {scope}model_string
245
+ def config[:envelope, :payload, :path]: "{model_uri}"
246
+ def export {{ config }}
247
+ """),
248
+ query_timeout_mins=remaining_timeout_minutes,
249
+ )
250
+ txn = response.transaction or {}
251
+ # The above `exec_raw` will throw an error if the transaction
252
+ # gets aborted. But in the case it gets cancelled, by the user
253
+ # or the system, it won't throw. In that case we also did not
254
+ # upload the input model.
255
+ if txn["state"] == "":
256
+ txn = solver.provider.resources.get_transaction(txn["id"]) or {}
257
+ if txn["state"] != "COMPLETED":
258
+ raise Exception(f"Transaction that materializes the solver inputs did not complete! ID: `{txn['id']}` State `{txn['state']}`")
259
+
260
+ # 2. Execute job and wait for completion.
261
+ remaining_timeout_minutes = calc_remaining_timeout_minutes(
262
+ start_time, query_timeout_mins, config_file_path=config_file_path
263
+ )
264
+ job_id = solver._exec_job(payload, log_to_console=log_to_console, query_timeout_mins=remaining_timeout_minutes)
265
+
266
+ # 3. Extract result.
267
+ remaining_timeout_minutes = calc_remaining_timeout_minutes(
268
+ start_time, query_timeout_mins, config_file_path=config_file_path
269
+ )
270
+ res = self.graph.exec_raw(
271
+ textwrap.dedent(f"""
272
+ ic result_not_empty("Solver result is empty.") requires not empty(result)
273
+
274
+ def result {{load_binary["snowflake://APP_STATE.RAI_INTERNAL_STAGE/job-results/{job_id}/result.binpb"] }}
275
+ def delete[:{scope}solve_output, :"i_{self.solve_index}"]: {scope}solve_output[:"i_{self.solve_index}"]
276
+
277
+ @no_diagnostics(:EXPERIMENTAL)
278
+ def insert[:{scope}solve_output, :"i_{self.solve_index}"]:
279
+ rel_primitive_solverlib_extract[result]
280
+
281
+ def output[:solver_error]: {scope}solve_output[:"i_{self.solve_index}", :error]
282
+ """),
283
+ readonly=False,
284
+ query_timeout_mins=remaining_timeout_minutes,
285
+ )
286
+ errors = []
287
+ for result in res.results:
288
+ if result["relationId"] == "/:output/:solver_error/String":
289
+ errors.extend(result["table"]["v1"])
290
+
291
+ # 4. Map results to solution.
292
+ with self.graph.rule(dynamic=True):
293
+ sol = self.Solution.add(index = self.solve_index)
294
+ for name in {"error", "termination_status", "solve_time_sec", "objective_value", "solver_version", "printed_model"}:
295
+ val = dsl.create_var()
296
+ getattr(self.solve_output, name)(val)
297
+ sol.set(**{name:val})
298
+
299
+ self.is_solved = True
300
+
301
+ if len(errors) > 0:
302
+ raise Exception("\n".join(errors))
303
+
304
+ return None
305
+
306
+ # Get scalar result information after solving.
307
+ def __getattr__(self, name:str):
308
+ if not self.is_solved:
309
+ raise Exception("Model has not been solved yet.")
310
+ if name in {"error", "termination_status", "solve_time_sec", "objective_value", "solver_version", "printed_model"}:
311
+ return getattr(self.Solution(index = self.solve_index), name)
312
+ else:
313
+ return None
314
+
315
+ # Get variable point values after solving. If `index` is specified, get the value
316
+ # of the variable in the return the `index`-th solution.
317
+ def value(self, var, index:int|None=None):
318
+ if not self.is_solved:
319
+ raise Exception("Model has not been solved yet.")
320
+ val = dsl.create_var()
321
+ unwrap_var = rel_sv.pyrel_unwrap(var)
322
+ if index:
323
+ self.solve_output.points(unwrap_var, index, val)
324
+ else:
325
+ self.solve_output.point(unwrap_var, val)
326
+ std.alias(val, "value")
327
+ return val
328
+
329
+ # --------------------------------------------------
330
+ # Operator definitions
331
+ # --------------------------------------------------
332
+
333
+ # Builtin binary operators
334
+
335
+ def plus(left, right):
336
+ return _make_fo_expr(10, left, right)
337
+
338
+ def minus(left, right):
339
+ return _make_fo_expr(11, left, right)
340
+
341
+ def mult(left, right):
342
+ return _make_fo_expr(12, left, right)
343
+
344
+ def div(left, right):
345
+ return _make_fo_expr(13, left, right)
346
+
347
+ def pow(left, right):
348
+ return _make_fo_expr(14, left, right)
349
+
350
+ def eq(left, right):
351
+ return _make_fo_expr(30, left, right)
352
+
353
+ def neq(left, right):
354
+ return _make_fo_expr(31, left, right)
355
+
356
+ def lte(left, right):
357
+ return _make_fo_expr(32, left, right)
358
+
359
+ def gte(left, right):
360
+ return _make_fo_expr(33, left, right)
361
+
362
+ def lt(left, right):
363
+ return _make_fo_expr(34, left, right)
364
+
365
+ def gt(left, right):
366
+ return _make_fo_expr(35, left, right)
367
+
368
+ # First order operators
369
+
370
+ def abs(arg):
371
+ return _make_fo_expr(20, arg)
372
+
373
+ def exp(arg):
374
+ return _make_fo_expr(21, arg)
375
+
376
+ def log(arg):
377
+ return _make_fo_expr(22, arg)
378
+
379
+ def integer(arg):
380
+ return _make_fo_expr(41, arg)
381
+
382
+ def zero_one(arg):
383
+ return _make_fo_expr(42, arg)
384
+
385
+ def interval(arg, low, high):
386
+ return _make_fo_expr(51, low, high, arg)
387
+
388
+ def integer_interval(arg, low, high):
389
+ return _make_fo_expr(50, low, high, 1, arg)
390
+
391
+ def if_then_else(cond, left, right):
392
+ return _make_fo_expr(60, cond, left, right)
393
+
394
+ def not_(arg):
395
+ return _make_fo_expr(61, arg)
396
+
397
+ def implies(left, right):
398
+ return _make_fo_expr(62, left, right)
399
+
400
+ def iff(left, right):
401
+ return _make_fo_expr(63, left, right)
402
+
403
+ def xor(left, right):
404
+ return _make_fo_expr(64, left, right)
405
+
406
+ def and_(*args):
407
+ return _make_fo_expr(70, *args)
408
+
409
+ def or_(*args):
410
+ return _make_fo_expr(71, *args)
411
+
412
+ # Aggregate operators
413
+
414
+ def sum(*args, per=[]) -> Any:
415
+ return _make_ho_expr(80, args, per)
416
+
417
+ def product(*args, per=[]) -> Any:
418
+ return _make_ho_expr(81, args, per)
419
+
420
+ def min(*args, per=[]) -> Any:
421
+ return _make_ho_expr(82, args, per)
422
+
423
+ def max(*args, per=[]) -> Any:
424
+ return _make_ho_expr(83, args, per)
425
+
426
+ def count(*args, per=[]) -> Any:
427
+ return _make_ho_expr(84, args, per)
428
+
429
+ def all_different(*args, per=[]) -> Any:
430
+ return _make_ho_expr(90, args, per)
431
+
432
+ # --------------------------------------------------
433
+ # Symbolic expression helpers
434
+ # --------------------------------------------------
435
+
436
+ def _make_fo_expr(*args):
437
+ expr = rel_sv.rel_primitive_solverlib_fo_appl(*args)
438
+ expr.__class__ = SolverExpression
439
+ return expr
440
+
441
+ # TODO(coey) test:
442
+ # dsl.tag(rel_sv.rel_primitive_solverlib_fo_appl, Builtins.Expensive)
443
+
444
+ def _make_ho_expr(op, args, per):
445
+ return SolverExpression(dsl.get_graph(), _ho_appl_def, [args, per, [op]])
446
+
447
+ _ho_appl_def = dsl.build.aggregate_def("_solverlib_ho_appl")
448
+
449
+ class SolverExpression(dsl.Expression):
450
+ def __init__(self, graph, op, args):
451
+ super().__init__(graph, op, args)
452
+
453
+ def _wrap_expr(e):
454
+ # If expression is not known to produce a serialized expression string,
455
+ # wrap it with the identity operation just in case
456
+ return e if isinstance(e, SolverExpression) else _make_fo_expr(0, e)
457
+
458
+ # Symbolic expression context, in which some builtin infix operators are redefined
459
+ # TODO(coey) handle comparison chains (e.g. 0 < x < y <= 1) or throw error
460
+ class Operators(dsl.Context):
461
+ def _supports_binary_op(self, op):
462
+ return op in _builtin_binary_map
463
+
464
+ def _make_binary_op(self, op, left, right):
465
+ return _make_fo_expr(_builtin_binary_map[op], left, right)
466
+
467
+ def operators():
468
+ return Operators(dsl.get_graph())
469
+
470
+ # Maps for Builtins operator to SolverLib operator ID
471
+ _builtin_binary_map = {
472
+ Builtins.plus: 10,
473
+ Builtins.minus: 11,
474
+ Builtins.mult: 12,
475
+ Builtins.div: 13,
476
+ Builtins.pow: 14,
477
+ Builtins.approx_eq: 30,
478
+ Builtins.neq: 31,
479
+ Builtins.lte: 32,
480
+ Builtins.gte: 33,
481
+ Builtins.lt: 34,
482
+ Builtins.gt: 35,
483
+ }
484
+
485
+ # Concatenate arguments into a string separated by underscores
486
+ def make_string(args:List):
487
+ string = args[0]
488
+ for arg in args[1:]:
489
+ string = rel_sv.concat(rel_sv.concat(string, "_"), arg)
490
+ return string
491
+
492
+
493
+ # --------------------------------------------------
494
+ # Solver
495
+ # --------------------------------------------------
496
+
497
+
498
+ @dataclass
499
+ class PollingState:
500
+ job_id: str
501
+ continuation_token: str
502
+ is_done: bool
503
+ log_to_console: bool
504
+
505
+
506
+ class Solver:
507
+ def __init__(
508
+ self,
509
+ solver_name: str,
510
+ engine_name: str | None = None,
511
+ engine_size: str | None = None,
512
+ auto_suspend_mins: int | None = None,
513
+ resources: SnowflakeResources | None = None,
514
+ ):
515
+ self.provider = Provider(resources=resources)
516
+ self.solver_name = solver_name.lower()
517
+
518
+ self.rai_config = self.provider.resources.config
519
+ settings: dict[str, Any] = {}
520
+ if "experimental" in self.rai_config:
521
+ exp_config = self.rai_config.get("experimental", {})
522
+ if isinstance(exp_config, dict):
523
+ if "solvers" in exp_config:
524
+ settings = exp_config["solvers"].copy()
525
+
526
+ # Engine configuration fields are not necessary for the solver
527
+ # settings so we `pop` them from the settings object. Default
528
+ # size and auto_suspend_mins are set in the `Provider` methods.
529
+ engine_name = engine_name or settings.pop("engine", None)
530
+ if not engine_name:
531
+ engine_name = self.provider.resources.get_user_based_engine_name()
532
+ self.engine_name = engine_name
533
+
534
+ self.engine_size = engine_size or settings.pop("engine_size", None)
535
+ self.engine_auto_suspend_mins = auto_suspend_mins or settings.pop("auto_suspend_mins", None)
536
+
537
+ # The settings are used when creating a solver engine, they
538
+ # may configure each individual solver.
539
+ self.engine_settings = settings
540
+
541
+ # Optimistically set the engine object to a `READY` engine to
542
+ # avoid checking the engine status on each execution.
543
+ self.engine:Optional[dict[str,Any]] = {"name": engine_name, "state": "READY"}
544
+
545
+ return None
546
+
547
+ # --------------------------------------------------
548
+ # Helper
549
+ # --------------------------------------------------
550
+ def _auto_create_solver_async(self):
551
+ name = self.engine_name
552
+ auto_suspend_mins = self.engine_auto_suspend_mins
553
+ size = self.engine_size
554
+ settings = self.engine_settings
555
+ with Spinner(
556
+ "Checking solver status",
557
+ leading_newline=True,
558
+ ) as spinner:
559
+ engine = None
560
+ engines = [e for e in self.provider.list_solvers() if e["name"] == name]
561
+ assert len(engines) == 1 or len(engines) == 0
562
+ if len(engines) != 0:
563
+ engine = engines[0]
564
+
565
+ if engine:
566
+ # TODO (dba) Logic engines support altering the
567
+ # auto_suspend_mins setting. Currently, we don't have
568
+ # this capability for solver engines, so users need to
569
+ # recreate or use another engine. For both the size
570
+ # and Gurobi configuration the user anyways has to
571
+ # create a new one as this configuration must happen
572
+ # when the engine is created.
573
+ settings_cannot_be_altered_msg = "The configuration of a solver engine happens when the engine is created and _cannot_ be changed. You either need to specify a new engine name or delete the current engine.\n\nSee `solvers.Provider().delete_solver()`."
574
+ # Make sure that the solver requested is enabled
575
+ # on the engine.
576
+ if self.solver_name not in engine["solvers"]:
577
+ raise Exception(
578
+ f"Solver `{self.solver_name}` is not enabled on `{name}`.\n\n" + settings_cannot_be_altered_msg
579
+ )
580
+
581
+ # Make sure size and auto_suspend_mins settings match
582
+ # what the user requests.
583
+ if size is not None and size != engine["size"]:
584
+ raise Exception(
585
+ f"Engine `{name}` has size setting of `{engine['size']}` but size is requested to be `{size}`.\n\n" + settings_cannot_be_altered_msg
586
+ )
587
+
588
+ if auto_suspend_mins is not None and auto_suspend_mins != engine["auto_suspend_mins"]:
589
+ raise Exception(
590
+ f"Engine `{name}` has auto_suspend_mins setting of `{engine['auto_suspend_mins']}` but auto_suspend_mins is requested to be `{auto_suspend_mins}`.\n\n" + settings_cannot_be_altered_msg
591
+ )
592
+
593
+ if engine["state"] == "PENDING":
594
+ spinner.update_messages(
595
+ {
596
+ "finished_message": f"Solver {name} is starting",
597
+ }
598
+ )
599
+ pass
600
+ elif engine["state"] == "SUSPENDED":
601
+ spinner.update_messages(
602
+ {
603
+ "finished_message": f"Resuming solver {name}",
604
+ }
605
+ )
606
+ self.provider.resume_solver_async(name)
607
+ elif engine["state"] == "READY":
608
+ spinner.update_messages(
609
+ {
610
+ "finished_message": f"Solver {name} is ready",
611
+ }
612
+ )
613
+ pass
614
+ else:
615
+ spinner.update_messages(
616
+ {
617
+ "message": f"Restarting solver {name}",
618
+ }
619
+ )
620
+ self.provider.delete_solver(name)
621
+ engine = None
622
+ if not engine:
623
+ # Validate Gurobi config.
624
+ if self.solver_name == "gurobi":
625
+ is_gurobi_configured = False
626
+ gurobi_config = settings.get("gurobi", {})
627
+ if all(
628
+ k in gurobi_config
629
+ for k in ["license_secret_name", "external_access_integration"]
630
+ ):
631
+ is_gurobi_configured = True
632
+ if not is_gurobi_configured:
633
+ raise Exception(
634
+ "Gurobi is not properly configured. You need to provide both `license_secret_name` and `external_access_integration` in its configuration, see https://docs.relational.ai/build/reasoners/prescriptive/solver-backends/gurobi/#usage"
635
+ )
636
+ self.provider.create_solver_async(name, settings=settings, size=size, auto_suspend_mins=auto_suspend_mins)
637
+ engine = self.provider.get_solver(name)
638
+ spinner.update_messages(
639
+ {
640
+ "finished_message": f"Starting solver {name}...",
641
+ }
642
+ )
643
+
644
+ self.engine = engine
645
+
646
+ def _exec_job(self, payload, log_to_console=True, query_timeout_mins: Optional[int]=None):
647
+ if self.engine is None:
648
+ raise Exception("Engine not initialized.")
649
+
650
+ with debugging.span("job") as job_span:
651
+ # Retry logic. If creating a job fails with an engine
652
+ # related error we will create/resume/... the engine and
653
+ # retry.
654
+ try:
655
+ job_id = self.provider.create_job_async(self.engine["name"], payload, query_timeout_mins=query_timeout_mins)
656
+ except Exception as e:
657
+ err_message = str(e).lower()
658
+ if isinstance(e, ResponseStatusException):
659
+ err_message = e.response.json().get("message", "")
660
+ if any(kw in err_message.lower() for kw in ENGINE_ERRORS + WORKER_ERRORS + ENGINE_NOT_READY_MSGS):
661
+ self._auto_create_solver_async()
662
+ # Wait until the engine is ready.
663
+ poll_with_specified_overhead(lambda: self._is_solver_ready(), 0.1)
664
+ job_id = self.provider.create_job_async(self.engine["name"], payload, query_timeout_mins=query_timeout_mins)
665
+ else:
666
+ raise e
667
+
668
+ job_span["job_id"] = job_id
669
+ debugging.event("job_created", job_span, job_id=job_id, engine_name=self.engine["name"], job_type=ENGINE_TYPE_SOLVER)
670
+ if not isinstance(job_id, str):
671
+ job_id = ""
672
+ polling_state = PollingState(job_id, "", False, log_to_console)
673
+
674
+ try:
675
+ with debugging.span("wait", job_id=job_id):
676
+ poll_with_specified_overhead(
677
+ lambda: self._check_job_status(polling_state), 0.1
678
+ )
679
+ except KeyboardInterrupt as e:
680
+ print(f"Canceling job {job_id}")
681
+ self.provider.cancel_job(job_id)
682
+ raise e
683
+
684
+ return job_id
685
+
686
+ def _is_solver_ready(self):
687
+ if self.engine is None:
688
+ raise Exception("Engine not initialized.")
689
+
690
+ result = self.provider.get_solver(self.engine["name"])
691
+
692
+ if result is None:
693
+ raise Exception("No engine available.")
694
+
695
+ self.engine = result
696
+ state = result["state"]
697
+ if state != "READY" and state != "PENDING":
698
+ # Might have suspended or otherwise gone. Recreate.
699
+ self._auto_create_solver_async()
700
+ return state == "READY"
701
+
702
+ def _check_job_status(self, state):
703
+ response = self.provider.get_job(state.job_id)
704
+ assert response, f"No results from get_job('{state.job_id}')"
705
+
706
+ status: str = response["state"]
707
+
708
+ self._print_solver_logs(state)
709
+
710
+ return status == "COMPLETED" or status == "FAILED" or status == "CANCELED"
711
+
712
+ def _print_solver_logs(self, state: PollingState):
713
+ if state.is_done:
714
+ return
715
+
716
+ resp = self.provider.get_job_events(state.job_id, state.continuation_token)
717
+
718
+ # Print solver logs to stdout.
719
+ for event in resp["events"]:
720
+ if event["type"] == "LogMessage":
721
+ if state.log_to_console:
722
+ print(event["event"]["message"])
723
+ else:
724
+ continue
725
+
726
+ state.continuation_token = resp["continuation_token"]
727
+ if state.continuation_token == "":
728
+ state.is_done = True
729
+
730
+
731
+ # --------------------------------------------------
732
+ # Provider
733
+ # --------------------------------------------------
734
+ #
735
+ # TODO (dba) We use an experimental and unified engine API for
736
+ # solvers. Once it is no longer experimental we can remove the
737
+ # provider here and use the normal PyRel provider.
738
+
739
+
740
+ class Provider:
741
+ def __init__(self, resources=None):
742
+ if not resources:
743
+ resources = v0.relationalai.Resources()
744
+ if not isinstance(resources, v0.relationalai.clients.snowflake.Resources):
745
+ raise Exception("Solvers are only supported on SPCS.")
746
+
747
+ self.resources = resources
748
+ self.direct_access_client: Optional[DirectAccessClient] = None
749
+
750
+ if isinstance(self.resources, DirectAccessResources):
751
+ self.direct_access_client = self.resources.direct_access_client
752
+
753
+ def create_solver(
754
+ self,
755
+ name: str,
756
+ size: str | None = None,
757
+ settings: dict | None = None,
758
+ auto_suspend_mins: int | None = None,
759
+ ):
760
+ if size is None:
761
+ size = "HIGHMEM_X64_S"
762
+ if settings is None:
763
+ settings = {}
764
+ engine_config: dict[str, Any] = {"settings": settings}
765
+ if auto_suspend_mins is not None:
766
+ engine_config["auto_suspend_mins"] = auto_suspend_mins
767
+ self.resources._exec_sql(
768
+ f"CALL {APP_NAME}.experimental.create_engine('{ENGINE_TYPE_SOLVER}', '{name}', '{size}', {engine_config});", None
769
+ )
770
+
771
+ def create_solver_async(
772
+ self,
773
+ name: str,
774
+ size: str | None = None,
775
+ settings: dict | None = None,
776
+ auto_suspend_mins: int | None = None,
777
+ ):
778
+ if size is None:
779
+ size = "HIGHMEM_X64_S"
780
+
781
+ if self.direct_access_client is not None:
782
+ payload:dict[str, Any] = {
783
+ "name": name,
784
+ "settings": settings,
785
+ }
786
+ if auto_suspend_mins is not None:
787
+ payload["auto_suspend_mins"] = auto_suspend_mins
788
+ if size is not None:
789
+ payload["size"] = size
790
+ response = self.direct_access_client.request(
791
+ "create_engine",
792
+ payload=payload,
793
+ path_params={"engine_type": "solver"},
794
+ )
795
+ if response.status_code != 200:
796
+ raise ResponseStatusException(
797
+ f"Failed to create engine {name} with size {size}.", response
798
+ )
799
+ else:
800
+ engine_config: dict[str, Any] = {}
801
+ if settings is not None:
802
+ engine_config["settings"] = settings
803
+ if auto_suspend_mins is not None:
804
+ engine_config["auto_suspend_mins"] = auto_suspend_mins
805
+ self.resources._exec_sql(
806
+ f"CALL {APP_NAME}.experimental.create_engine_async('{ENGINE_TYPE_SOLVER}', '{name}', '{size}', {engine_config});",
807
+ None
808
+ )
809
+
810
+ def delete_solver(self, name: str):
811
+ if self.direct_access_client is not None:
812
+ response = self.direct_access_client.request(
813
+ "delete_engine", path_params = {"engine_type": ENGINE_TYPE_SOLVER, "engine_name": name}
814
+ )
815
+ if response.status_code != 200:
816
+ raise ResponseStatusException("Failed to delete engine.", response)
817
+ return None
818
+ else:
819
+ self.resources._exec_sql(
820
+ f"CALL {APP_NAME}.experimental.delete_engine('{ENGINE_TYPE_SOLVER}', '{name}');",
821
+ None
822
+ )
823
+
824
+ def resume_solver_async(self, name: str):
825
+ if self.direct_access_client is not None:
826
+ response = self.direct_access_client.request(
827
+ "resume_engine", path_params = {"engine_type": ENGINE_TYPE_SOLVER, "engine_name": name}
828
+ )
829
+ if response.status_code != 200:
830
+ raise ResponseStatusException("Failed to resume engine.", response)
831
+ return None
832
+ else:
833
+ self.resources._exec_sql(
834
+ f"CALL {APP_NAME}.experimental.resume_engine_async('{ENGINE_TYPE_SOLVER}', '{name}');",
835
+ None
836
+ )
837
+ return None
838
+
839
+ def get_solver(self, name: str):
840
+ if self.direct_access_client is not None:
841
+ response = self.direct_access_client.request(
842
+ "get_engine", path_params = {"engine_type": ENGINE_TYPE_SOLVER, "engine_name": name}
843
+ )
844
+ if response.status_code != 200:
845
+ raise ResponseStatusException("Failed to get engine.", response)
846
+ solver = response.json()
847
+ if not solver :
848
+ return None
849
+ solver_state = {
850
+ "name": solver["name"],
851
+ "id": solver["id"],
852
+ "size": solver["size"],
853
+ "state": solver["status"], # callers are expecting 'state'
854
+ "created_by": solver["created_by"],
855
+ "created_on": solver["created_on"],
856
+ "updated_on": solver["updated_on"],
857
+ "version": solver["version"],
858
+ "auto_suspend": solver["auto_suspend_mins"],
859
+ "suspends_at": solver["suspends_at"],
860
+ "solvers": []
861
+ if solver["settings"] == ""
862
+ else [
863
+ k
864
+ for (k,v) in json.loads(solver["settings"]).items()
865
+ if isinstance(v, dict) and v.get("enabled", False)
866
+ ],
867
+ }
868
+ return solver_state
869
+ else:
870
+ results = self.resources._exec_sql(
871
+ f"CALL {APP_NAME}.experimental.get_engine('{ENGINE_TYPE_SOLVER}', '{name}');",
872
+ None
873
+ )
874
+ return solver_list_to_dicts(results)[0]
875
+
876
+ def list_solvers(self, state: str | None = None):
877
+ if self.direct_access_client is not None:
878
+ response = self.direct_access_client.request(
879
+ "list_engines"
880
+ )
881
+ if response.status_code != 200:
882
+ raise ResponseStatusException("Failed to list engines.", response)
883
+ response_content = response.json()
884
+ if not response_content:
885
+ return []
886
+ engines = [
887
+ {
888
+ "name": engine["name"],
889
+ "id": engine["id"],
890
+ "size": engine["size"],
891
+ "state": engine["status"], # callers are expecting 'state'
892
+ "created_by": engine["created_by"],
893
+ "created_on": engine["created_on"],
894
+ "updated_on": engine["updated_on"],
895
+ "auto_suspend_mins": engine["auto_suspend_mins"],
896
+ "solvers": []
897
+ if engine["settings"] == ""
898
+ else [
899
+ k
900
+ for (k, v) in json.loads(engine["settings"]).items()
901
+ if isinstance(v, dict) and v.get("enabled", False)
902
+ ],
903
+ }
904
+ for engine in response_content.get("engines", [])
905
+ if (state is None or engine.get("status") == state) and (engine.get("type") == ENGINE_TYPE_SOLVER)
906
+ ]
907
+ return sorted(engines, key=lambda x: x["name"])
908
+ else:
909
+ where_clause = f"WHERE TYPE='{ENGINE_TYPE_SOLVER}'"
910
+ where_clause = (
911
+ f"{where_clause} AND STATUS = '{state.upper()}'" if state else where_clause
912
+ )
913
+ statement = f"SELECT NAME,ID,SIZE,STATUS,CREATED_BY,CREATED_ON,UPDATED_ON,AUTO_SUSPEND_MINS,SETTINGS FROM {APP_NAME}.experimental.engines {where_clause};"
914
+ results = self.resources._exec_sql(statement, None)
915
+ return solver_list_to_dicts(results)
916
+
917
+ # --------------------------------------------------
918
+ # Job API
919
+ # --------------------------------------------------
920
+
921
+ def create_job_async(self, engine_name, payload, query_timeout_mins: Optional[int]=None):
922
+ payload_json = json.dumps(payload)
923
+
924
+ if query_timeout_mins is None and (timeout_value := self.resources.config.get("query_timeout_mins", DEFAULT_QUERY_TIMEOUT_MINS)) is not None:
925
+ query_timeout_mins = int(timeout_value)
926
+
927
+ if self.direct_access_client is not None:
928
+ job = {
929
+ "job_type":ENGINE_TYPE_SOLVER,
930
+ "worker_name": engine_name,
931
+ "timeout_mins": query_timeout_mins,
932
+ "payload": payload_json,
933
+ }
934
+ response = self.direct_access_client.request(
935
+ "create_job",
936
+ payload=job,
937
+ )
938
+ if response.status_code != 200:
939
+ raise ResponseStatusException("Failed to create job.", response)
940
+ response_content = response.json()
941
+ return response_content["id"]
942
+ else:
943
+ if query_timeout_mins is not None:
944
+ sql_string = textwrap.dedent(f"""
945
+ CALL {APP_NAME}.experimental.exec_job_async('{ENGINE_TYPE_SOLVER}', '{engine_name}', '{payload_json}', null, {query_timeout_mins})
946
+ """)
947
+ else:
948
+ sql_string = textwrap.dedent(f"""
949
+ CALL {APP_NAME}.experimental.exec_job_async('{ENGINE_TYPE_SOLVER}', '{engine_name}', '{payload_json}')
950
+ """)
951
+ res = self.resources._exec_sql(sql_string, None)
952
+ return res[0]["ID"]
953
+
954
+ def list_jobs(self, state=None, limit=None):
955
+ if self.direct_access_client is not None:
956
+ response = self.direct_access_client.request(
957
+ "list_jobs"
958
+ )
959
+ if response.status_code != 200:
960
+ raise ResponseStatusException("Failed to list jobs.", response)
961
+ response_content = response.json()
962
+ if not response_content:
963
+ return []
964
+ jobs = [
965
+ {
966
+ "id": job["id"],
967
+ "state": job["state"],
968
+ "created_by": job["created_by"],
969
+ "created_on": job["created_on"],
970
+ "finished_at": job.get("finished_at", None),
971
+ "duration": job["duration"] if "duration" in job else 0,
972
+ "solver": json.loads(job["payload"]).get("solver", ""),
973
+ "engine": job.get("engine_name", job["worker_name"]),
974
+ }
975
+ for job in response_content.get("jobs", [])
976
+ if state is None or job.get("state") == state
977
+ ]
978
+ return sorted(jobs, key=lambda x: x["created_on"], reverse=True)
979
+ else:
980
+ state_clause = f"AND STATE = '{state.upper()}'" if state else ""
981
+ limit_clause = f"LIMIT {limit}" if limit else ""
982
+ results = self.resources._exec_sql(
983
+ f"SELECT ID,STATE,CREATED_BY,CREATED_ON,FINISHED_AT,DURATION,PAYLOAD,ENGINE_NAME FROM {APP_NAME}.experimental.jobs where type='{ENGINE_TYPE_SOLVER}' {state_clause} ORDER BY created_on DESC {limit_clause};",
984
+ None
985
+ )
986
+ return job_list_to_dicts(results)
987
+
988
+ def get_job(self, id: str):
989
+ if self.direct_access_client is not None:
990
+ response = self.direct_access_client.request(
991
+ "get_job", path_params = {"job_type": ENGINE_TYPE_SOLVER, "job_id": id}
992
+ )
993
+ if response.status_code != 200:
994
+ raise ResponseStatusException("Failed to get job.", response)
995
+ response_content = response.json()
996
+ return response_content["job"]
997
+ else:
998
+ results = self.resources._exec_sql(
999
+ f"CALL {APP_NAME}.experimental.get_job('{ENGINE_TYPE_SOLVER}', '{id}');",
1000
+ None
1001
+ )
1002
+ return job_list_to_dicts(results)[0]
1003
+
1004
+ def get_job_events(self, job_id: str, continuation_token: str = ""):
1005
+ if self.direct_access_client is not None:
1006
+ response = self.direct_access_client.request(
1007
+ "get_job_events",
1008
+ path_params = {"job_type": ENGINE_TYPE_SOLVER, "job_id": job_id, "stream_name": "progress"},
1009
+ query_params={"continuation_token": continuation_token},
1010
+ )
1011
+ if response.status_code != 200:
1012
+ raise ResponseStatusException("Failed to get job events.", response)
1013
+ response_content = response.json()
1014
+ if not response_content:
1015
+ return {
1016
+ "events": [],
1017
+ "continuation_token": None
1018
+ }
1019
+ return response_content
1020
+ else:
1021
+ results = self.resources._exec_sql(
1022
+ f"SELECT {APP_NAME}.experimental.get_job_events('{ENGINE_TYPE_SOLVER}', '{job_id}', '{continuation_token}');",
1023
+ None
1024
+ )
1025
+ if not results:
1026
+ return {"events": [], "continuation_token": None}
1027
+ row = results[0][0]
1028
+ if not isinstance(row, str):
1029
+ row = ""
1030
+ return json.loads(row)
1031
+
1032
+ def cancel_job(self, id: str):
1033
+ if self.direct_access_client is not None:
1034
+ response = self.direct_access_client.request(
1035
+ "cancel_job", path_params = {"job_type": ENGINE_TYPE_SOLVER, "job_id": id}
1036
+ )
1037
+ if response.status_code != 200:
1038
+ raise ResponseStatusException("Failed to cancel job.", response)
1039
+ return None
1040
+ else:
1041
+ self.resources._exec_sql(
1042
+ f"CALL {APP_NAME}.experimental.cancel_job('{ENGINE_TYPE_SOLVER}', '{id}');",
1043
+ None
1044
+ )
1045
+ return None
1046
+
1047
+
1048
+ def solver_list_to_dicts(results):
1049
+ if not results:
1050
+ return []
1051
+ return [
1052
+ {
1053
+ "name": row["NAME"],
1054
+ "id": row["ID"],
1055
+ "size": row["SIZE"],
1056
+ "state": row["STATUS"], # callers are expecting 'state'
1057
+ "created_by": row["CREATED_BY"],
1058
+ "created_on": row["CREATED_ON"],
1059
+ "updated_on": row["UPDATED_ON"],
1060
+ "auto_suspend_mins": row["AUTO_SUSPEND_MINS"],
1061
+ "solvers": []
1062
+ if row["SETTINGS"] == ""
1063
+ else [
1064
+ k
1065
+ for (k, v) in json.loads(row["SETTINGS"]).items()
1066
+ if isinstance(v, dict) and v.get("enabled", False)
1067
+ ],
1068
+ }
1069
+ for row in results
1070
+ ]
1071
+
1072
+ def job_list_to_dicts(results):
1073
+ if not results:
1074
+ return []
1075
+ return [
1076
+ {
1077
+ "id": row["ID"],
1078
+ "state": row["STATE"],
1079
+ "created_by": row["CREATED_BY"],
1080
+ "created_on": row["CREATED_ON"],
1081
+ "finished_at": row["FINISHED_AT"],
1082
+ "duration": row["DURATION"] if "DURATION" in row else 0,
1083
+ "solver": json.loads(row["PAYLOAD"])["solver"],
1084
+ "engine": row["ENGINE_NAME"],
1085
+ }
1086
+ for row in results
1087
+ ]