jade-lang 0.1.0

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 (326) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +23 -0
  3. data/LICENSE +21 -0
  4. data/README.md +386 -0
  5. data/exe/jade +6 -0
  6. data/lib/jade/ast/node.rb +44 -0
  7. data/lib/jade/ast/nodes.rb +35 -0
  8. data/lib/jade/ast/pretty_printer.rb +50 -0
  9. data/lib/jade/ast.rb +723 -0
  10. data/lib/jade/calendar/runtime.rb +15 -0
  11. data/lib/jade/cli/fmt.rb +96 -0
  12. data/lib/jade/cli/lsp.rb +13 -0
  13. data/lib/jade/cli/q.rb +113 -0
  14. data/lib/jade/cli.rb +43 -0
  15. data/lib/jade/clock/runtime.rb +13 -0
  16. data/lib/jade/codegen/boundary/cache.rb +94 -0
  17. data/lib/jade/codegen/boundary/specialized/list.rb +65 -0
  18. data/lib/jade/codegen/boundary/specialized/maybe.rb +40 -0
  19. data/lib/jade/codegen/boundary/specialized/record.rb +165 -0
  20. data/lib/jade/codegen/boundary/specialized/scalar.rb +67 -0
  21. data/lib/jade/codegen/boundary/specialized.rb +106 -0
  22. data/lib/jade/codegen/boundary.rb +189 -0
  23. data/lib/jade/codegen/constructor_reference.rb +18 -0
  24. data/lib/jade/codegen/context.rb +96 -0
  25. data/lib/jade/codegen/emitter.rb +81 -0
  26. data/lib/jade/codegen/function_call.rb +367 -0
  27. data/lib/jade/codegen/function_declaration.rb +199 -0
  28. data/lib/jade/codegen/helpers.rb +103 -0
  29. data/lib/jade/codegen/implementation.rb +178 -0
  30. data/lib/jade/codegen/inline.rb +89 -0
  31. data/lib/jade/codegen/inlines.rb +326 -0
  32. data/lib/jade/codegen/method_names.rb +54 -0
  33. data/lib/jade/codegen/pattern/constructor.rb +57 -0
  34. data/lib/jade/codegen/port_decoder.rb +77 -0
  35. data/lib/jade/codegen/pretty.rb +53 -0
  36. data/lib/jade/codegen/transforms/fold_shape.rb +222 -0
  37. data/lib/jade/codegen/transforms/self_call.rb +80 -0
  38. data/lib/jade/codegen/transforms/tail_call.rb +120 -0
  39. data/lib/jade/codegen/variant_declaration.rb +41 -0
  40. data/lib/jade/codegen.rb +400 -0
  41. data/lib/jade/compiler.rb +69 -0
  42. data/lib/jade/decode.rb +320 -0
  43. data/lib/jade/diagnostics/renderer.rb +121 -0
  44. data/lib/jade/diagnostics.rb +77 -0
  45. data/lib/jade/did_you_mean.rb +16 -0
  46. data/lib/jade/entry.rb +177 -0
  47. data/lib/jade/error.rb +72 -0
  48. data/lib/jade/formatter/accesses.rb +37 -0
  49. data/lib/jade/formatter/bindings.rb +29 -0
  50. data/lib/jade/formatter/body.rb +50 -0
  51. data/lib/jade/formatter/calls.rb +51 -0
  52. data/lib/jade/formatter/case_of.rb +31 -0
  53. data/lib/jade/formatter/case_of_branch.rb +59 -0
  54. data/lib/jade/formatter/collections.rb +78 -0
  55. data/lib/jade/formatter/declarations.rb +178 -0
  56. data/lib/jade/formatter/exposing.rb +48 -0
  57. data/lib/jade/formatter/function_declaration.rb +72 -0
  58. data/lib/jade/formatter/helper.rb +122 -0
  59. data/lib/jade/formatter/if_then_else.rb +64 -0
  60. data/lib/jade/formatter/infix_application.rb +69 -0
  61. data/lib/jade/formatter/lambda.rb +50 -0
  62. data/lib/jade/formatter/leaves.rb +111 -0
  63. data/lib/jade/formatter/module_node.rb +26 -0
  64. data/lib/jade/formatter/pattern.rb +61 -0
  65. data/lib/jade/formatter/type.rb +67 -0
  66. data/lib/jade/formatter.rb +38 -0
  67. data/lib/jade/frontend/comment_attacher.rb +121 -0
  68. data/lib/jade/frontend/desugaring/placeholder.rb +39 -0
  69. data/lib/jade/frontend/desugaring/resolved.rb +63 -0
  70. data/lib/jade/frontend/desugaring.rb +217 -0
  71. data/lib/jade/frontend/fixity_fixer.rb +209 -0
  72. data/lib/jade/frontend/forward_declaration/body.rb +30 -0
  73. data/lib/jade/frontend/forward_declaration/error/bad_import.rb +19 -0
  74. data/lib/jade/frontend/forward_declaration/error/exposed_type_not_found.rb +18 -0
  75. data/lib/jade/frontend/forward_declaration/error/exposed_value_not_found.rb +18 -0
  76. data/lib/jade/frontend/forward_declaration/error/module_not_found.rb +18 -0
  77. data/lib/jade/frontend/forward_declaration/error/private_type_expansion.rb +19 -0
  78. data/lib/jade/frontend/forward_declaration/error/tuple_arity_overflow.rb +25 -0
  79. data/lib/jade/frontend/forward_declaration/error/type_not_found.rb +29 -0
  80. data/lib/jade/frontend/forward_declaration/error/type_not_lowerable.rb +16 -0
  81. data/lib/jade/frontend/forward_declaration/error/unknown_extends_interface.rb +18 -0
  82. data/lib/jade/frontend/forward_declaration/error.rb +11 -0
  83. data/lib/jade/frontend/forward_declaration/function_declaration.rb +32 -0
  84. data/lib/jade/frontend/forward_declaration/helper.rb +91 -0
  85. data/lib/jade/frontend/forward_declaration/implementation.rb +63 -0
  86. data/lib/jade/frontend/forward_declaration/implementation_function.rb +39 -0
  87. data/lib/jade/frontend/forward_declaration/import_declaration.rb +115 -0
  88. data/lib/jade/frontend/forward_declaration/interface_declaration.rb +66 -0
  89. data/lib/jade/frontend/forward_declaration/interop_import_declaration.rb +99 -0
  90. data/lib/jade/frontend/forward_declaration/module.rb +98 -0
  91. data/lib/jade/frontend/forward_declaration/struct_declaration.rb +42 -0
  92. data/lib/jade/frontend/forward_declaration/type_declaration.rb +42 -0
  93. data/lib/jade/frontend/forward_declaration.rb +71 -0
  94. data/lib/jade/frontend/pattern_analysis/exhaustiveness.rb +65 -0
  95. data/lib/jade/frontend/pattern_analysis/matrix.rb +235 -0
  96. data/lib/jade/frontend/pattern_analysis.rb +40 -0
  97. data/lib/jade/frontend/semantic_analysis/assign.rb +20 -0
  98. data/lib/jade/frontend/semantic_analysis/body.rb +33 -0
  99. data/lib/jade/frontend/semantic_analysis/case_of.rb +19 -0
  100. data/lib/jade/frontend/semantic_analysis/case_of_branch.rb +20 -0
  101. data/lib/jade/frontend/semantic_analysis/char_literal.rb +14 -0
  102. data/lib/jade/frontend/semantic_analysis/constructor_reference.rb +64 -0
  103. data/lib/jade/frontend/semantic_analysis/error/circular_extends.rb +19 -0
  104. data/lib/jade/frontend/semantic_analysis/error/constant_not_callable.rb +24 -0
  105. data/lib/jade/frontend/semantic_analysis/error/constructor_not_found.rb +34 -0
  106. data/lib/jade/frontend/semantic_analysis/error/constructor_pattern_arity_mismatch.rb +24 -0
  107. data/lib/jade/frontend/semantic_analysis/error/duplicate_field.rb +18 -0
  108. data/lib/jade/frontend/semantic_analysis/error/duplicate_function_declaration.rb +25 -0
  109. data/lib/jade/frontend/semantic_analysis/error/duplicate_record_field.rb +19 -0
  110. data/lib/jade/frontend/semantic_analysis/error/invalid_list_rest_pattern.rb +21 -0
  111. data/lib/jade/frontend/semantic_analysis/error/kwargs_on_non_constructor.rb +17 -0
  112. data/lib/jade/frontend/semantic_analysis/error/missing_exposing_clause.rb +17 -0
  113. data/lib/jade/frontend/semantic_analysis/error/missing_extends_implementation.rb +21 -0
  114. data/lib/jade/frontend/semantic_analysis/error/missing_field.rb +20 -0
  115. data/lib/jade/frontend/semantic_analysis/error/missing_implementation_function.rb +19 -0
  116. data/lib/jade/frontend/semantic_analysis/error/module_not_found.rb +22 -0
  117. data/lib/jade/frontend/semantic_analysis/error/nested_task_port.rb +19 -0
  118. data/lib/jade/frontend/semantic_analysis/error/non_task_port.rb +22 -0
  119. data/lib/jade/frontend/semantic_analysis/error/orphan_implementation.rb +20 -0
  120. data/lib/jade/frontend/semantic_analysis/error/predicate_must_return_bool.rb +22 -0
  121. data/lib/jade/frontend/semantic_analysis/error/predicate_name_not_allowed.rb +25 -0
  122. data/lib/jade/frontend/semantic_analysis/error/shadowing_error.rb +22 -0
  123. data/lib/jade/frontend/semantic_analysis/error/type_args_mismatch.rb +25 -0
  124. data/lib/jade/frontend/semantic_analysis/error/type_param_required.rb +19 -0
  125. data/lib/jade/frontend/semantic_analysis/error/unbound_type_variable.rb +22 -0
  126. data/lib/jade/frontend/semantic_analysis/error/undefined_variable.rb +29 -0
  127. data/lib/jade/frontend/semantic_analysis/error/unknown_field.rb +20 -0
  128. data/lib/jade/frontend/semantic_analysis/error/unknown_implementation_function.rb +19 -0
  129. data/lib/jade/frontend/semantic_analysis/error/unused_interface_type_param.rb +24 -0
  130. data/lib/jade/frontend/semantic_analysis/error/value_not_exposed.rb +23 -0
  131. data/lib/jade/frontend/semantic_analysis/error/variable_not_found.rb +25 -0
  132. data/lib/jade/frontend/semantic_analysis/error.rb +40 -0
  133. data/lib/jade/frontend/semantic_analysis/function_call.rb +60 -0
  134. data/lib/jade/frontend/semantic_analysis/function_declaration.rb +58 -0
  135. data/lib/jade/frontend/semantic_analysis/grouping.rb +17 -0
  136. data/lib/jade/frontend/semantic_analysis/helper.rb +152 -0
  137. data/lib/jade/frontend/semantic_analysis/if_then_else.rb +20 -0
  138. data/lib/jade/frontend/semantic_analysis/implementation.rb +143 -0
  139. data/lib/jade/frontend/semantic_analysis/implementation_function.rb +16 -0
  140. data/lib/jade/frontend/semantic_analysis/import_declaration.rb +14 -0
  141. data/lib/jade/frontend/semantic_analysis/interface_declaration.rb +45 -0
  142. data/lib/jade/frontend/semantic_analysis/interop_import_declaration.rb +69 -0
  143. data/lib/jade/frontend/semantic_analysis/keyed_call/validation.rb +109 -0
  144. data/lib/jade/frontend/semantic_analysis/keyed_call.rb +88 -0
  145. data/lib/jade/frontend/semantic_analysis/lambda.rb +23 -0
  146. data/lib/jade/frontend/semantic_analysis/list.rb +17 -0
  147. data/lib/jade/frontend/semantic_analysis/literal.rb +23 -0
  148. data/lib/jade/frontend/semantic_analysis/member_access.rb +87 -0
  149. data/lib/jade/frontend/semantic_analysis/module_node.rb +27 -0
  150. data/lib/jade/frontend/semantic_analysis/pattern_binding.rb +27 -0
  151. data/lib/jade/frontend/semantic_analysis/pattern_constructor.rb +47 -0
  152. data/lib/jade/frontend/semantic_analysis/pattern_list.rb +33 -0
  153. data/lib/jade/frontend/semantic_analysis/pattern_literal.rb +17 -0
  154. data/lib/jade/frontend/semantic_analysis/pattern_record.rb +25 -0
  155. data/lib/jade/frontend/semantic_analysis/pattern_wildcard.rb +14 -0
  156. data/lib/jade/frontend/semantic_analysis/qualified_access.rb +14 -0
  157. data/lib/jade/frontend/semantic_analysis/record_access.rb +14 -0
  158. data/lib/jade/frontend/semantic_analysis/record_field.rb +17 -0
  159. data/lib/jade/frontend/semantic_analysis/record_literal.rb +21 -0
  160. data/lib/jade/frontend/semantic_analysis/record_update.rb +21 -0
  161. data/lib/jade/frontend/semantic_analysis/struct_declaration.rb +44 -0
  162. data/lib/jade/frontend/semantic_analysis/tuple.rb +17 -0
  163. data/lib/jade/frontend/semantic_analysis/type_declaration.rb +69 -0
  164. data/lib/jade/frontend/semantic_analysis/variable_reference.rb +27 -0
  165. data/lib/jade/frontend/semantic_analysis/variant_declaration.rb +18 -0
  166. data/lib/jade/frontend/semantic_analysis.rb +161 -0
  167. data/lib/jade/frontend/type_checking/canonicalize.rb +97 -0
  168. data/lib/jade/frontend/type_checking/constraints/deriving/decodable.rb +144 -0
  169. data/lib/jade/frontend/type_checking/constraints/deriving/encodable.rb +144 -0
  170. data/lib/jade/frontend/type_checking/constraints/deriving/eq.rb +265 -0
  171. data/lib/jade/frontend/type_checking/constraints/deriving/helpers.rb +59 -0
  172. data/lib/jade/frontend/type_checking/constraints/deriving.rb +28 -0
  173. data/lib/jade/frontend/type_checking/constraints.rb +101 -0
  174. data/lib/jade/frontend/type_checking/definition.rb +71 -0
  175. data/lib/jade/frontend/type_checking/env.rb +79 -0
  176. data/lib/jade/frontend/type_checking/error/case_of_branches_type_mismatch.rb +19 -0
  177. data/lib/jade/frontend/type_checking/error/derivation_failed.rb +21 -0
  178. data/lib/jade/frontend/type_checking/error/function_body_type_mismatch.rb +23 -0
  179. data/lib/jade/frontend/type_checking/error/function_call_type_mismatch.rb +37 -0
  180. data/lib/jade/frontend/type_checking/error/if_branch_type_mismatch.rb +19 -0
  181. data/lib/jade/frontend/type_checking/error/if_branches_type_mismatch.rb +18 -0
  182. data/lib/jade/frontend/type_checking/error/if_condition_type_mismatch.rb +17 -0
  183. data/lib/jade/frontend/type_checking/error/implementation_type_mismatch.rb +20 -0
  184. data/lib/jade/frontend/type_checking/error/list_item_type_mismatch.rb +19 -0
  185. data/lib/jade/frontend/type_checking/error/missing_implementation.rb +20 -0
  186. data/lib/jade/frontend/type_checking/error/missing_patterns.rb +26 -0
  187. data/lib/jade/frontend/type_checking/error/pattern_type_mismatch.rb +13 -0
  188. data/lib/jade/frontend/type_checking/error/port_not_decodable.rb +38 -0
  189. data/lib/jade/frontend/type_checking/error/record_access_type_mismatch.rb +14 -0
  190. data/lib/jade/frontend/type_checking/error/type_mismatch.rb +23 -0
  191. data/lib/jade/frontend/type_checking/error/unresolved_constraint.rb +20 -0
  192. data/lib/jade/frontend/type_checking/error.rb +18 -0
  193. data/lib/jade/frontend/type_checking/expected.rb +23 -0
  194. data/lib/jade/frontend/type_checking/generalization.rb +17 -0
  195. data/lib/jade/frontend/type_checking/generalizer.rb +38 -0
  196. data/lib/jade/frontend/type_checking/inference/assign.rb +58 -0
  197. data/lib/jade/frontend/type_checking/inference/body.rb +45 -0
  198. data/lib/jade/frontend/type_checking/inference/case_of.rb +102 -0
  199. data/lib/jade/frontend/type_checking/inference/constructor_reference.rb +21 -0
  200. data/lib/jade/frontend/type_checking/inference/function_call.rb +132 -0
  201. data/lib/jade/frontend/type_checking/inference/function_declaration.rb +70 -0
  202. data/lib/jade/frontend/type_checking/inference/grouping.rb +18 -0
  203. data/lib/jade/frontend/type_checking/inference/helpers.rb +34 -0
  204. data/lib/jade/frontend/type_checking/inference/if_then_else.rb +46 -0
  205. data/lib/jade/frontend/type_checking/inference/implementation.rb +150 -0
  206. data/lib/jade/frontend/type_checking/inference/import_declaration.rb +19 -0
  207. data/lib/jade/frontend/type_checking/inference/interface_declaration.rb +18 -0
  208. data/lib/jade/frontend/type_checking/inference/interop_import_declaration.rb +18 -0
  209. data/lib/jade/frontend/type_checking/inference/lambda.rb +87 -0
  210. data/lib/jade/frontend/type_checking/inference/list.rb +52 -0
  211. data/lib/jade/frontend/type_checking/inference/literal.rb +24 -0
  212. data/lib/jade/frontend/type_checking/inference/module.rb +18 -0
  213. data/lib/jade/frontend/type_checking/inference/pattern.rb +135 -0
  214. data/lib/jade/frontend/type_checking/inference/qualified_access.rb +23 -0
  215. data/lib/jade/frontend/type_checking/inference/record_access.rb +35 -0
  216. data/lib/jade/frontend/type_checking/inference/record_field.rb +19 -0
  217. data/lib/jade/frontend/type_checking/inference/record_literal.rb +24 -0
  218. data/lib/jade/frontend/type_checking/inference/record_update.rb +37 -0
  219. data/lib/jade/frontend/type_checking/inference/struct_declaration.rb +18 -0
  220. data/lib/jade/frontend/type_checking/inference/type_declaration.rb +18 -0
  221. data/lib/jade/frontend/type_checking/inference/variable_reference.rb +27 -0
  222. data/lib/jade/frontend/type_checking/inference.rb +27 -0
  223. data/lib/jade/frontend/type_checking/instantiation.rb +24 -0
  224. data/lib/jade/frontend/type_checking/loader.rb +80 -0
  225. data/lib/jade/frontend/type_checking/placeholder.rb +12 -0
  226. data/lib/jade/frontend/type_checking/port_resolution.rb +123 -0
  227. data/lib/jade/frontend/type_checking/result.rb +41 -0
  228. data/lib/jade/frontend/type_checking/scheme.rb +20 -0
  229. data/lib/jade/frontend/type_checking/state.rb +52 -0
  230. data/lib/jade/frontend/type_checking/substitution.rb +93 -0
  231. data/lib/jade/frontend/type_checking/unification.rb +282 -0
  232. data/lib/jade/frontend/type_checking/var_gen.rb +33 -0
  233. data/lib/jade/frontend/type_checking.rb +129 -0
  234. data/lib/jade/frontend/unused_analysis.rb +41 -0
  235. data/lib/jade/frontend/usage_analysis/reference_index.rb +53 -0
  236. data/lib/jade/frontend/usage_analysis.rb +195 -0
  237. data/lib/jade/frontend.rb +101 -0
  238. data/lib/jade/interop/boundary.rb +68 -0
  239. data/lib/jade/interop/error.rb +84 -0
  240. data/lib/jade/interop/lowering/error.rb +32 -0
  241. data/lib/jade/interop/lowering.rb +53 -0
  242. data/lib/jade/interop/runtime.rb +24 -0
  243. data/lib/jade/interop.rb +1 -0
  244. data/lib/jade/lexer.rb +189 -0
  245. data/lib/jade/lsp/converters.rb +542 -0
  246. data/lib/jade/lsp/handlers.rb +340 -0
  247. data/lib/jade/lsp/server.rb +63 -0
  248. data/lib/jade/lsp/snippets.rb +100 -0
  249. data/lib/jade/lsp/state.rb +25 -0
  250. data/lib/jade/lsp.rb +16 -0
  251. data/lib/jade/module_loader/cache.rb +56 -0
  252. data/lib/jade/module_loader/dependency_graph.rb +23 -0
  253. data/lib/jade/module_loader/dependency_resolver.rb +48 -0
  254. data/lib/jade/module_loader/normalize.rb +34 -0
  255. data/lib/jade/module_loader/topological_sort.rb +41 -0
  256. data/lib/jade/module_loader.rb +127 -0
  257. data/lib/jade/parsing/combinators.rb +291 -0
  258. data/lib/jade/parsing/error.rb +154 -0
  259. data/lib/jade/parsing/token.rb +12 -0
  260. data/lib/jade/parsing/type.rb +92 -0
  261. data/lib/jade/parsing.rb +674 -0
  262. data/lib/jade/port.rb +1 -0
  263. data/lib/jade/registry.rb +79 -0
  264. data/lib/jade/result.rb +121 -0
  265. data/lib/jade/runtime.rb +127 -0
  266. data/lib/jade/source.rb +62 -0
  267. data/lib/jade/stdlib/basics.rb +214 -0
  268. data/lib/jade/stdlib/bytes.rb +70 -0
  269. data/lib/jade/stdlib/calendar.rb +405 -0
  270. data/lib/jade/stdlib/char.rb +27 -0
  271. data/lib/jade/stdlib/clock.rb +342 -0
  272. data/lib/jade/stdlib/compiled.rb +48 -0
  273. data/lib/jade/stdlib/decode/params.rb +154 -0
  274. data/lib/jade/stdlib/decode.rb +315 -0
  275. data/lib/jade/stdlib/dict.rb +134 -0
  276. data/lib/jade/stdlib/encode.rb +143 -0
  277. data/lib/jade/stdlib/intrinsics.rb +280 -0
  278. data/lib/jade/stdlib/list.rb +214 -0
  279. data/lib/jade/stdlib/maybe.rb +73 -0
  280. data/lib/jade/stdlib/result.rb +131 -0
  281. data/lib/jade/stdlib/set.rb +123 -0
  282. data/lib/jade/stdlib/string.rb +65 -0
  283. data/lib/jade/stdlib/task.rb +55 -0
  284. data/lib/jade/stdlib/tuple.rb +21 -0
  285. data/lib/jade/stdlib.rb +112 -0
  286. data/lib/jade/symbol/anonymous_record.rb +7 -0
  287. data/lib/jade/symbol/base.rb +15 -0
  288. data/lib/jade/symbol/constructor.rb +11 -0
  289. data/lib/jade/symbol/derived_function.rb +5 -0
  290. data/lib/jade/symbol/function.rb +15 -0
  291. data/lib/jade/symbol/function_type.rb +7 -0
  292. data/lib/jade/symbol/implementation.rb +17 -0
  293. data/lib/jade/symbol/implementation_template.rb +13 -0
  294. data/lib/jade/symbol/interface.rb +11 -0
  295. data/lib/jade/symbol/interface_function.rb +18 -0
  296. data/lib/jade/symbol/interop_function.rb +22 -0
  297. data/lib/jade/symbol/lambda.rb +7 -0
  298. data/lib/jade/symbol/parser.rb +79 -0
  299. data/lib/jade/symbol/partial_application.rb +7 -0
  300. data/lib/jade/symbol/record_type.rb +8 -0
  301. data/lib/jade/symbol/stdlib_function.rb +15 -0
  302. data/lib/jade/symbol/stdlib_implementation.rb +7 -0
  303. data/lib/jade/symbol/struct.rb +15 -0
  304. data/lib/jade/symbol/type_application.rb +8 -0
  305. data/lib/jade/symbol/type_ref.rb +11 -0
  306. data/lib/jade/symbol/union.rb +15 -0
  307. data/lib/jade/symbol/value_ref.rb +15 -0
  308. data/lib/jade/symbol/variable.rb +7 -0
  309. data/lib/jade/symbol/variant.rb +11 -0
  310. data/lib/jade/symbol.rb +162 -0
  311. data/lib/jade/task.rb +103 -0
  312. data/lib/jade/tasks/rspec.rb +266 -0
  313. data/lib/jade/tasks.rb +70 -0
  314. data/lib/jade/type/anonymous_record.rb +33 -0
  315. data/lib/jade/type/application.rb +21 -0
  316. data/lib/jade/type/base.rb +9 -0
  317. data/lib/jade/type/constraint.rb +17 -0
  318. data/lib/jade/type/constructor.rb +19 -0
  319. data/lib/jade/type/function.rb +18 -0
  320. data/lib/jade/type/partial_application.rb +17 -0
  321. data/lib/jade/type/unit.rb +15 -0
  322. data/lib/jade/type/var.rb +21 -0
  323. data/lib/jade/type.rb +259 -0
  324. data/lib/jade/version.rb +3 -0
  325. data/lib/jade.rb +55 -0
  326. metadata +387 -0
@@ -0,0 +1,79 @@
1
+ require 'jade/module_loader'
2
+ require 'jade/entry'
3
+
4
+ module Jade
5
+ Registry = Data.define(:modules, :implementations, :source_root, :dependency_graph, :overlays) do
6
+ def initialize(modules: {}, implementations: {}, source_root: nil, dependency_graph: nil, overlays: {})
7
+ super(
8
+ modules:,
9
+ implementations:,
10
+ source_root:,
11
+ dependency_graph: dependency_graph || ModuleLoader::DependencyGraph.new,
12
+ overlays:,
13
+ )
14
+ end
15
+
16
+ def self.entry(name)
17
+ Entry.empty(name)
18
+ end
19
+
20
+ def modules_in_topo_order
21
+ ModuleLoader::TopologicalSort
22
+ .sort(dependency_graph)
23
+ .map { get(it) }
24
+ end
25
+
26
+ def get(module_name)
27
+ modules.dig(module_name)
28
+ end
29
+
30
+ def update_module(entry)
31
+ fail 'cannot update entry that does not exist' unless modules[entry.name]
32
+
33
+ with(
34
+ modules: modules.merge(entry.name => entry),
35
+ implementations: implementations.merge(entry.implementations),
36
+ )
37
+ end
38
+
39
+ def add_module(entry)
40
+ next_registry = with(
41
+ modules: modules.merge(entry.name => entry),
42
+ implementations: implementations.merge(entry.implementations),
43
+ )
44
+
45
+ if entry.ast
46
+ ModuleLoader::DependencyResolver.resolve(entry, next_registry)
47
+ else
48
+ next_registry
49
+ end
50
+ end
51
+
52
+ def add_dependencies(entry, imports)
53
+ with(dependency_graph: dependency_graph.add(entry.name, imports))
54
+ end
55
+
56
+ def find_node_at(uri, offset)
57
+ modules
58
+ .each_value
59
+ .find { it.source&.uri == uri }
60
+ &.ast
61
+ &.find_at(offset)
62
+ end
63
+
64
+ def lookup(symbol)
65
+ *module_parts, name = symbol.qualified_name.split('.')
66
+ module_entry = module_parts.join('.').then { modules[it] }
67
+
68
+ case symbol
69
+ in Symbol::ValueRef
70
+ module_entry.values[name]
71
+
72
+ in Symbol::TypeRef
73
+ module_entry.types[name]
74
+ end
75
+ end
76
+ end
77
+
78
+ ImportEntry = Data.define(:module_name, :alias, :unqualified_symbols, :qualified_symbols)
79
+ end
@@ -0,0 +1,121 @@
1
+ module Jade
2
+ Ok = Data.define(:ok) do
3
+ private :ok
4
+
5
+ def initialize(ok: nil)
6
+ super
7
+ end
8
+
9
+ def map
10
+ Ok.new(yield(ok))
11
+ end
12
+ alias_method :map_both, :map
13
+
14
+ def and_then
15
+ yield(ok)
16
+ end
17
+
18
+ def and_tap
19
+ result = yield(ok)
20
+ case result
21
+ in Ok then self
22
+ in Err then result
23
+ end
24
+ end
25
+
26
+ def map_error
27
+ self
28
+ end
29
+
30
+ def map2(result)
31
+ case result
32
+ in Ok(ok2) then Ok.new(yield(ok, ok2))
33
+ in Err(_) then result
34
+ end
35
+ end
36
+
37
+ def and_then_combine
38
+ result = yield(ok)
39
+ case result
40
+ in Ok(ok2) then Ok.new([*ok, ok2])
41
+ in Err(_) then result
42
+ end
43
+ end
44
+
45
+ def on_err(_error_type = nil)
46
+ self
47
+ end
48
+
49
+ def with_default(_default)
50
+ ok
51
+ end
52
+
53
+ def ok?
54
+ true
55
+ end
56
+
57
+ def error?
58
+ false
59
+ end
60
+ end
61
+
62
+ module Results
63
+ def self.sequence(results)
64
+ results.reduce(Ok[[]]) do |acc, r|
65
+ acc.and_then { |list| r.map { list + [it] } }
66
+ end
67
+ end
68
+ end
69
+
70
+ Err = Data.define(:err) do
71
+ private :err
72
+
73
+ def map
74
+ self
75
+ end
76
+
77
+ def and_then
78
+ self
79
+ end
80
+
81
+ def map_error
82
+ Err.new(yield(err))
83
+ end
84
+ alias_method :map_both, :map_error
85
+
86
+ def map2(_)
87
+ self
88
+ end
89
+
90
+ def and_tap
91
+ self
92
+ end
93
+
94
+ def and_then_combine
95
+ self
96
+ end
97
+
98
+ def on_err(error_type = nil)
99
+ return yield(err) unless error_type
100
+
101
+ case err
102
+ when error_type
103
+ yield(err)
104
+ else
105
+ self
106
+ end
107
+ end
108
+
109
+ def with_default(default)
110
+ default
111
+ end
112
+
113
+ def ok?
114
+ false
115
+ end
116
+
117
+ def error?
118
+ true
119
+ end
120
+ end
121
+ end
@@ -0,0 +1,127 @@
1
+ require 'base64'
2
+
3
+ require 'jade/interop/runtime'
4
+ require 'jade/decode'
5
+ require 'jade/interop/boundary'
6
+
7
+ module Jade
8
+ module Tuple
9
+ Tuple2 = Data.define(:_1, :_2) do
10
+ def to_s = "(#{[_1, _2].map(&:to_s).join(', ')})"
11
+ end
12
+
13
+ Tuple3 = Data.define(:_1, :_2, :_3) do
14
+ def to_s = "(#{[_1, _2, _3].map(&:to_s).join(', ')})"
15
+ end
16
+
17
+ Tuple4 = Data.define(:_1, :_2, :_3, :_4) do
18
+ def to_s = "(#{[_1, _2, _3, _4].map(&:to_s).join(', ')})"
19
+ end
20
+ end
21
+
22
+ module Basics
23
+ GT = Data.define()
24
+ EQ = Data.define()
25
+ LT = Data.define()
26
+ end
27
+
28
+ # Compiled code calls `a.compare(b)`; natives ship `<=>` not `compare`.
29
+ # Guarded so a future Ruby or third-party `compare` wins — clobbering
30
+ # would silently swap semantics.
31
+ [Integer, Float, ::String].each do |klass|
32
+ next if klass.method_defined?(:compare)
33
+
34
+ klass.class_eval do
35
+ define_method(:compare) do |other|
36
+ case self <=> other
37
+ when -1 then Basics::LT[]
38
+ when 0 then Basics::EQ[]
39
+ when 1 then Basics::GT[]
40
+ end
41
+ end
42
+ end
43
+ end
44
+
45
+ module Bytes
46
+ Bytes = Data.define(:bin) do
47
+ def to_s = "Bytes(#{bin.bytesize})"
48
+ def +(other) = Bytes.new(bin + other.bin)
49
+ end
50
+ end
51
+
52
+ module Dict
53
+ Dict = Data.define(:hash) do
54
+ def to_s
55
+ pairs = hash.map { |k, v| "#{k}: #{v}" }.join(', ')
56
+ "Dict(#{pairs})"
57
+ end
58
+ end
59
+ end
60
+
61
+ module Set
62
+ Set = Data.define(:hash) do
63
+ def to_s = "Set(#{hash.keys.join(', ')})"
64
+ end
65
+ end
66
+
67
+ module Runtime
68
+ extend self
69
+ extend Interop::Runtime
70
+
71
+ INTRINSICS = {}
72
+ IMPLEMENTATIONS = {}
73
+ IMPL_CACHE = {}
74
+ RECORD_CLASSES = {}
75
+ @booted = false
76
+
77
+ def boot!
78
+ return if @booted
79
+ @booted = true
80
+
81
+ require "jade/stdlib/basics"
82
+ require "jade/stdlib/string"
83
+ require "jade/stdlib/list"
84
+ require "jade/stdlib/tuple"
85
+ require "jade/stdlib/task"
86
+ require "jade/stdlib/decode"
87
+ require "jade/stdlib/encode"
88
+ require "jade/stdlib/bytes"
89
+ require "jade/stdlib/dict"
90
+ require "jade/stdlib/set"
91
+ end
92
+
93
+ def intr(name)
94
+ boot!
95
+ INTRINSICS[name] || fail("Intrinsic #{name} does not exist")
96
+ end
97
+
98
+ def register(name, &block)
99
+ INTRINSICS[name] = block
100
+ end
101
+
102
+ # Memoized class for anonymous record literals. Without this, every
103
+ # `{a: 1, b: 2}` expression evaluated in a hot loop would call
104
+ # `Data.define(:a, :b)` and allocate a fresh anonymous class, defeating
105
+ # YJIT's inline cache on every subsequent property access.
106
+ def record(*keys)
107
+ RECORD_CLASSES[keys] ||= Data.define(*keys)
108
+ end
109
+
110
+ def register_impl(interface_name, ruby_class, functions)
111
+ IMPLEMENTATIONS[[interface_name, ruby_class]] = functions
112
+ IMPL_CACHE.clear
113
+ end
114
+
115
+ # Cached per [interface_name, value.class]; `register_impl` invalidates.
116
+ def impl_for(interface_name, value)
117
+ boot!
118
+ key = [interface_name, value.class]
119
+ IMPL_CACHE[key] ||= begin
120
+ raw = IMPLEMENTATIONS[key] || fail("No implementation of #{interface_name} for #{value.class}")
121
+ raw.any? { |_, v| v.is_a?(::String) } \
122
+ ? raw.transform_values { |v| v.is_a?(::String) ? intr(v) : v }
123
+ : raw
124
+ end
125
+ end
126
+ end
127
+ end
@@ -0,0 +1,62 @@
1
+ module Jade
2
+ Source = Data.define(:uri, :text, :line_starts) do
3
+ def self.load(source_root, uri, overlays: {})
4
+ text = overlays[uri] || File.read(File.join(source_root, uri))
5
+ new(uri, text)
6
+ end
7
+
8
+ def self.load_from_module_name(source_root, name, overlays: {})
9
+ name.split('.')
10
+ .compact
11
+ .map { snake_case(it) }
12
+ .then { |(*rest, last)| rest + [last + '.jd'] }
13
+ .then { File.join(*it) }
14
+ .then { load(resolve_root(source_root, it, name), it, overlays:) }
15
+ end
16
+
17
+ def self.resolve_root(source_root, uri, name)
18
+ return source_root if File.exist?(File.join(source_root, uri))
19
+
20
+ candidates = Jade.extensions.select { |r| File.exist?(File.join(r, uri)) }
21
+
22
+ case candidates.size
23
+ when 0 then source_root
24
+ when 1 then candidates.first
25
+ else raise "Module #{name} is provided by multiple extensions: #{candidates.join(', ')}"
26
+ end
27
+ end
28
+
29
+ def initialize(uri:, text:, line_starts: calculate_line_starts(text))
30
+ super
31
+ end
32
+
33
+ def to_module_name
34
+ uri
35
+ .delete_suffix('.jd')
36
+ .split('/')
37
+ .map { Source.camelize(it) }
38
+ .join('.')
39
+ end
40
+
41
+ private
42
+
43
+ # Byte offsets — must match StringScanner's byte-positioned `pos`,
44
+ # which is what every AST `:range` carries. Using char positions
45
+ # here drifts spans by the multi-byte-char count beyond the first
46
+ # non-ASCII glyph in the file.
47
+ def calculate_line_starts(text)
48
+ [0] + text.b.enum_for(:scan, /\n/).map { Regexp.last_match.end(0) }
49
+ end
50
+
51
+ def self.camelize(str)
52
+ str.split('_').map { |part| part.capitalize }.join
53
+ end
54
+
55
+ def self.snake_case(str)
56
+ str
57
+ .gsub(/([A-Z]+)([A-Z][a-z])/, '\1_\2') # ABCXyz -> ABC_Xyz
58
+ .gsub(/([a-z\d])([A-Z])/, '\1_\2') # abcXyz -> abc_Xyz
59
+ .downcase
60
+ end
61
+ end
62
+ end
@@ -0,0 +1,214 @@
1
+ require 'jade/stdlib/intrinsics'
2
+
3
+ module Jade
4
+ module Stdlib
5
+ module Basics
6
+ extend Intrinsics
7
+
8
+ union :Never
9
+ union :Int
10
+ union :Float
11
+ union :Bool
12
+
13
+ native_type :Int, Integer
14
+ native_type :Float, ::Float
15
+ native_type :Bool, TrueClass, FalseClass
16
+
17
+ union :Ordering
18
+ variant :GT, of: :Ordering
19
+ variant :EQ, of: :Ordering
20
+ variant :LT, of: :Ordering
21
+
22
+ interface(
23
+ 'Eq',
24
+ 'a',
25
+ { '(==)' => 'a, a -> Bool' },
26
+ )
27
+
28
+ implementation('Eq', 'Int', '(==)' => 'int_eq')
29
+ implementation('Eq', 'Float', '(==)' => 'float_eq')
30
+ implementation('Eq', 'Bool', '(==)' => 'bool_eq')
31
+
32
+ interface(
33
+ 'Comparable',
34
+ 'a',
35
+ { 'compare' => 'a, a -> Ordering' },
36
+ )
37
+
38
+ implementation('Comparable', 'Int', 'compare' => 'int_compare')
39
+ implementation('Comparable', 'Float', 'compare' => 'float_compare')
40
+
41
+ interface(
42
+ 'Numeric',
43
+ 'a',
44
+ { '(+)' => 'a, a -> a', '(-)' => 'a, a -> a', '(*)' => 'a, a -> a', '(/)' => 'a, a -> a' }
45
+ )
46
+
47
+ implementation('Numeric', 'Int', '(+)' => 'int_add', '(-)' => 'int_sub', '(*)' => 'int_mul', '(/)' => 'int_div')
48
+ implementation('Numeric', 'Float', '(+)' => 'float_add', '(-)' => 'float_sub', '(*)' => 'float_mul', '(/)' => 'float_div')
49
+
50
+ interface(
51
+ 'Appendable',
52
+ 'a',
53
+ { '(++)' => 'a, a -> a' },
54
+ )
55
+
56
+ interface(
57
+ 'Mappable',
58
+ 'f',
59
+ { 'map' => 'f(a), (a -> b) -> f(b)' },
60
+ )
61
+
62
+ interface(
63
+ 'Chainable',
64
+ 'f',
65
+ { 'and_then' => 'f(a), (a -> f(b)) -> f(b)' },
66
+ )
67
+
68
+ function(
69
+ '(!=)',
70
+ { one: 'a', other: 'a' },
71
+ 'Bool',
72
+ constraints: [['Basics.Eq', 'a']],
73
+ body: Symbol::DerivedFunction.new(
74
+ params: ['one', 'other'],
75
+ body: [:!, [:call, [:impl_arg, 0, '(==)'], [[:var, 'one'], [:var, 'other']]]],
76
+ ),
77
+ )
78
+
79
+ function('int_add', { a: 'Int', b: 'Int' }, 'Int')
80
+ function('int_sub', { a: 'Int', b: 'Int' }, 'Int')
81
+ function('int_mul', { a: 'Int', b: 'Int' }, 'Int')
82
+ function('int_div', { a: 'Int', b: 'Int' }, 'Int')
83
+ function('mod', { a: 'Int', b: 'Int' }, 'Int')
84
+
85
+ function('float_add', { a: 'Float', b: 'Float' }, 'Float')
86
+ function('float_sub', { a: 'Float', b: 'Float' }, 'Float')
87
+ function('float_mul', { a: 'Float', b: 'Float' }, 'Float')
88
+ function('float_div', { a: 'Float', b: 'Float' }, 'Float')
89
+
90
+ function(:to_float, { n: 'Int' }, 'Float')
91
+ function(:floor, { n: 'Float' }, 'Int')
92
+ function(:ceiling, { n: 'Float' }, 'Int')
93
+ function(:round, { n: 'Float' }, 'Int')
94
+ function(:truncate, { n: 'Float' }, 'Int')
95
+
96
+ function('identity', { a: 'a' }, 'a')
97
+ function('always', { x: 'a' }, 'b -> a')
98
+ function('not', { a: 'Bool' }, 'Bool')
99
+
100
+ function(
101
+ '(<)',
102
+ { a: 'a', b: 'a' },
103
+ 'Bool',
104
+ constraints: [['Basics.Comparable', 'a']],
105
+ body: Symbol::DerivedFunction.new(
106
+ params: ['a', 'b'],
107
+ body: [:case,
108
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
109
+ [
110
+ [[:constructor, 'Basics.LT', []], [true]],
111
+ [[:_], [false]],
112
+ ],
113
+ ],
114
+ ),
115
+ )
116
+
117
+ function(
118
+ '(>)',
119
+ { a: 'a', b: 'a' },
120
+ 'Bool',
121
+ constraints: [['Basics.Comparable', 'a']],
122
+ body: Symbol::DerivedFunction.new(
123
+ params: ['a', 'b'],
124
+ body: [:case,
125
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
126
+ [
127
+ [[:constructor, 'Basics.GT', []], [true]],
128
+ [[:_], [false]],
129
+ ],
130
+ ],
131
+ ),
132
+ )
133
+
134
+ function(
135
+ '(<=)',
136
+ { a: 'a', b: 'a' },
137
+ 'Bool',
138
+ constraints: [['Basics.Comparable', 'a']],
139
+ body: Symbol::DerivedFunction.new(
140
+ params: ['a', 'b'],
141
+ body: [:case,
142
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
143
+ [
144
+ [[:constructor, 'Basics.GT', []], [false]],
145
+ [[:_], [true]],
146
+ ],
147
+ ],
148
+ ),
149
+ )
150
+
151
+ function(
152
+ '(>=)',
153
+ { a: 'a', b: 'a' },
154
+ 'Bool',
155
+ constraints: [['Basics.Comparable', 'a']],
156
+ body: Symbol::DerivedFunction.new(
157
+ params: ['a', 'b'],
158
+ body: [:case,
159
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
160
+ [
161
+ [[:constructor, 'Basics.LT', []], [false]],
162
+ [[:_], [true]],
163
+ ],
164
+ ],
165
+ ),
166
+ )
167
+
168
+ function(
169
+ 'min',
170
+ { a: 'a', b: 'a' },
171
+ 'a',
172
+ constraints: [['Basics.Comparable', 'a']],
173
+ body: Symbol::DerivedFunction.new(
174
+ params: ['a', 'b'],
175
+ body: [:case,
176
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
177
+ [
178
+ [[:constructor, 'Basics.GT', []], [[:var, 'b']]],
179
+ [[:_], [[:var, 'a']]],
180
+ ],
181
+ ],
182
+ ),
183
+ )
184
+
185
+ function(
186
+ 'max',
187
+ { a: 'a', b: 'a' },
188
+ 'a',
189
+ constraints: [['Basics.Comparable', 'a']],
190
+ body: Symbol::DerivedFunction.new(
191
+ params: ['a', 'b'],
192
+ body: [:case,
193
+ [:call, [:impl_arg, 0, 'compare'], [[:var, 'a'], [:var, 'b']]],
194
+ [
195
+ [[:constructor, 'Basics.LT', []], [[:var, 'b']]],
196
+ [[:_], [[:var, 'a']]],
197
+ ],
198
+ ],
199
+ ),
200
+ )
201
+
202
+ function('(&&)', { a: 'Bool', b: 'Bool' }, 'Bool')
203
+ function('(||)', { a: 'Bool', b: 'Bool' }, 'Bool')
204
+
205
+ default_importing :*
206
+
207
+ function('int_eq', { one: 'Int', other: 'Int' }, 'Bool')
208
+ function('float_eq', { one: 'Float', other: 'Float' }, 'Bool')
209
+ function('bool_eq', { one: 'Bool', other: 'Bool' }, 'Bool')
210
+ function('int_compare', { a: 'Int', b: 'Int' }, 'Ordering')
211
+ function('float_compare', { a: 'Float', b: 'Float' }, 'Ordering')
212
+ end
213
+ end
214
+ end
@@ -0,0 +1,70 @@
1
+ require 'base64'
2
+ require 'jade/stdlib/intrinsics'
3
+
4
+ module Jade
5
+ module Stdlib
6
+ module Bytes
7
+ extend Intrinsics
8
+
9
+ import Basics
10
+ import Maybe
11
+ import List
12
+
13
+ union :Bytes
14
+
15
+ native_type :Bytes, Jade::Bytes::Bytes
16
+
17
+ implementation('Eq', 'Bytes', '(==)' => 'bytes_eq')
18
+ implementation('Appendable', 'Bytes', '(++)' => 'bytes_append')
19
+
20
+ function(:empty, {}, 'Bytes')
21
+ function(:width, { bytes: 'Bytes' }, 'Int')
22
+
23
+ function(
24
+ :from_list,
25
+ { ints: 'List(Int)' },
26
+ 'Maybe(Bytes)',
27
+ ) do |ints|
28
+ if ints.all? { it.is_a?(Integer) && it.between?(0, 255) }
29
+ Jade::Maybe::Just[Jade::Bytes::Bytes[ints.pack('C*')]]
30
+ else
31
+ Jade::Maybe::Nothing[]
32
+ end
33
+ end
34
+
35
+ function(:to_list, { bytes: 'Bytes' }, 'List(Int)')
36
+ function(:from_string, { s: 'String' }, 'Bytes')
37
+ function(:to_string, { bytes: 'Bytes' }, 'Maybe(String)')
38
+
39
+ function(
40
+ :from_hex,
41
+ { s: 'String' },
42
+ 'Maybe(Bytes)',
43
+ ) do |s|
44
+ if s.length.even? && s.match?(/\A[0-9a-fA-F]*\z/)
45
+ Jade::Maybe::Just[Jade::Bytes::Bytes[[s].pack('H*')]]
46
+ else
47
+ Jade::Maybe::Nothing[]
48
+ end
49
+ end
50
+
51
+ function(:to_hex, { bytes: 'Bytes' }, 'String')
52
+ function(:to_base64_url, { bytes: 'Bytes' }, 'String')
53
+
54
+ function(
55
+ :from_base64_url,
56
+ { s: 'String' },
57
+ 'Maybe(Bytes)',
58
+ ) do |s|
59
+ Jade::Maybe::Just[Jade::Bytes::Bytes[::Base64.urlsafe_decode64(s)]]
60
+ rescue ::ArgumentError
61
+ Jade::Maybe::Nothing[]
62
+ end
63
+
64
+ default_importing('Bytes')
65
+
66
+ function('bytes_eq', { a: 'Bytes', b: 'Bytes' }, 'Bool')
67
+ function('bytes_append', { a: 'Bytes', b: 'Bytes' }, 'Bytes')
68
+ end
69
+ end
70
+ end