From 13323c55a4d34ddb74b67fab684d4431f9624dd1 Mon Sep 17 00:00:00 2001 From: Eduardo Julian Date: Mon, 29 Nov 2021 17:53:01 -0400 Subject: New packaging for Ruby programs. --- lux-ruby/source/program.lux | 58 +++++---- .../library/lux/tool/compiler/meta/packager.lux | 3 +- .../lux/tool/compiler/meta/packager/ruby.lux | 140 +++++++++++++++++++++ stdlib/source/program/aedifex/command/build.lux | 7 +- stdlib/source/program/aedifex/command/test.lux | 50 ++++---- stdlib/source/program/compositor.lux | 14 ++- 6 files changed, 216 insertions(+), 56 deletions(-) create mode 100644 stdlib/source/library/lux/tool/compiler/meta/packager/ruby.lux diff --git a/lux-ruby/source/program.lux b/lux-ruby/source/program.lux index 9aa71067f..3a55ccafc 100644 --- a/lux-ruby/source/program.lux +++ b/lux-ruby/source/program.lux @@ -67,7 +67,7 @@ [meta [archive {"+" Archive}] ["[0]" packager "_" - ["[1]" script]]]]]]] + ["[1]" ruby]]]]]]] [program ["/" compositor ["/[0]" cli] @@ -870,9 +870,16 @@ (def: (program context program) (Program _.Expression _.Statement) - (_.statement (_.apply_lambda/* (list (runtime.lux//program_args _.command_line_arguments) - _.nil) - program))) + (let [normal_runtime? (_.do "const_defined?" + (list (_.string (_.code _.command_line_arguments))) + {.#None} + (_.local "Object"))] + (_.statement + (_.apply_lambda/* (list (runtime.lux//program_args (_.? normal_runtime? + _.command_line_arguments + (_.array (list)))) + _.nil) + program)))) (def: (declare_success! _) (-> Any (Async Any)) @@ -880,27 +887,24 @@ (program: [service /cli.service] (let [extension ".rb"] - (exec (do async.monad - [_ (/.compiler [/static.#host @.ruby - /static.#host_module_extension extension - /static.#target (/cli.target service) - /static.#artifact_extension extension] - ..expander - analysis.bundle - ..platform - generation.bundle - (function.constant extension/bundle.empty) - ..program - [Register _.Expression _.Statement] - ..extender - service - [(packager.package (: _.Statement (_.manual "")) - _.code - _.then - (|>>)) - (format (/cli.target service) - (# file.default separator) - "program" - extension)])] - (..declare_success! [])) + (exec + (do async.monad + [_ (/.compiler [/static.#host @.ruby + /static.#host_module_extension extension + /static.#target (/cli.target service) + /static.#artifact_extension extension] + ..expander + analysis.bundle + ..platform + generation.bundle + (function.constant extension/bundle.empty) + ..program + [Register _.Expression _.Statement] + ..extender + service + [packager.package + (format (/cli.target service) + (# file.default separator) + "program")])] + (..declare_success! [])) (io.io [])))) diff --git a/stdlib/source/library/lux/tool/compiler/meta/packager.lux b/stdlib/source/library/lux/tool/compiler/meta/packager.lux index d57f52400..5d6fe712e 100644 --- a/stdlib/source/library/lux/tool/compiler/meta/packager.lux +++ b/stdlib/source/library/lux/tool/compiler/meta/packager.lux @@ -26,7 +26,8 @@ (-> (Dictionary file.Path Binary) Archive Context - (Try Binary))) + (Try (Either Binary + (List [Text Binary]))))) (type: .public Order (List [archive.ID (List artifact.ID)])) diff --git a/stdlib/source/library/lux/tool/compiler/meta/packager/ruby.lux b/stdlib/source/library/lux/tool/compiler/meta/packager/ruby.lux new file mode 100644 index 000000000..a375a908a --- /dev/null +++ b/stdlib/source/library/lux/tool/compiler/meta/packager/ruby.lux @@ -0,0 +1,140 @@ +(.using + [library + [lux {"-" Module} + [type {"+" :sharing}] + [abstract + ["[0]" monad {"+" do}]] + [control + ["[0]" try {"+" Try}]] + [data + [binary {"+" Binary}] + ["[0]" product] + ["[0]" text + ["%" format {"+" format}] + [encoding + ["[0]" utf8]]] + [collection + ["[0]" sequence] + ["[0]" list ("[1]#[0]" functor mix)] + ["[0]" dictionary {"+" Dictionary}] + ["[0]" set {"+" Set}]]] + [math + [number + ["[0]" nat]]] + [target + ["_" ruby]] + [world + ["[0]" file]]]] + [program + [compositor + ["[0]" static {"+" Static}]]] + ["[0]" // {"+" Packager} + [// + ["[0]" archive {"+" Output} + ["[0]" descriptor {"+" Module Descriptor}] + ["[0]" artifact] + ["[0]" document {"+" Document}] + ["tree_shaking" dependency]] + [cache + ["[0]" dependency {"+" Order}]] + ["[0]" io "_" + ["[1]" archive]] + [// + [language + ["$" lux + [generation {"+" Context}]]]]]]) + +(def: (bundle_module module module_id necessary_dependencies output) + (-> Module archive.ID (Set Context) Output (Try (Maybe _.Statement))) + (do [! try.monad] + [] + (case (|> output + sequence.list + (list.only (function (_ [artifact_id custom content]) + (set.member? necessary_dependencies [module_id artifact_id])))) + {.#End} + (in {.#None}) + + artifacts + (do ! + [bundle (monad.mix ! + (function (_ [artifact custom_name content] so_far) + (|> content + (# utf8.codec decoded) + (# ! each + (|>> :expected + (:sharing [directive] + directive + so_far + + directive) + (_.then so_far))))) + (_.comment "Lux module" + (_.statement (_.string ""))) + artifacts)] + (in {.#Some bundle}))))) + +(def: module_file + (-> archive.ID file.Path) + (|>> %.nat (text.suffix ".rb"))) + +(def: (write_module mapping necessary_dependencies [module [module_id [descriptor document output]]] sink) + (-> (Dictionary Module archive.ID) (Set Context) + [Module [archive.ID [Descriptor (Document .Module) Output]]] + (List [archive.ID [Text Binary]]) + (Try (List [archive.ID [Text Binary]]))) + (do [! try.monad] + [bundle (: (Try (Maybe _.Statement)) + (..bundle_module module module_id necessary_dependencies output))] + (case bundle + {.#None} + (in sink) + + {.#Some bundle} + (let [entry_content (|> (list) + (list#mix _.then bundle) + (: _.Statement) + _.code + (# utf8.codec encoded))] + (in (list& [module_id [(..module_file module_id) entry_content]] + sink)))))) + +(def: .public main_file + "main.rb") + +(def: module_id_mapping + (-> Order (Dictionary Module archive.ID)) + (|>> (list#each (function (_ [module [module_id [descriptor document output]]]) + [module module_id])) + (dictionary.of_list text.hash))) + +(def: included_modules + (All (_ a) (-> (List [archive.ID a]) (Set archive.ID))) + (|>> (list#each product.left) + (list#mix set.has (set.empty nat.hash)))) + +(def: .public (package host_dependencies archive program) + Packager + (do [! try.monad] + [.let [necessary_dependencies (tree_shaking.necessary_dependencies archive)] + order (dependency.load_order $.key archive) + entries (monad.mix ! (..write_module (module_id_mapping order) necessary_dependencies) {.#End} order) + .let [included_modules (..included_modules entries) + imports (|> order + (list.only (|>> product.right product.left (set.member? included_modules))) + list.reversed + (list#each (function (_ [module [module_id [descriptor document output]]]) + (let [relative_path (_.do "gsub" (list (_.string main_file) + (_.string (..module_file module_id))) + {.#None} + (_.local "__FILE__"))] + (_.statement (_.require/1 relative_path))))) + (list#mix _.then (_.comment "Lux program" + (_.statement (_.string "")))) + (: _.Statement) + _.code + (# utf8.codec encoded))]] + (in (|> entries + (list#each product.right) + {.#Item [..main_file imports]} + {.#Right})))) diff --git a/stdlib/source/program/aedifex/command/build.lux b/stdlib/source/program/aedifex/command/build.lux index 9fbc34542..15a671615 100644 --- a/stdlib/source/program/aedifex/command/build.lux +++ b/stdlib/source/program/aedifex/command/build.lux @@ -23,6 +23,11 @@ [number ["n" nat] ["i" int]]] + [tool + [compiler + [meta + ["[0]" packager + ["[0]_[1]" ruby]]]]] [world ["[0]" program {"+" Program}] ["[0]" file {"+" Path}] @@ -291,7 +296,7 @@ ([#JS ///.#js "program.js"] [#Python ///.#java "program.py"] [#Lua ///.#java "program.lua"] - [#Ruby ///.#java "program.rb"])) + [#Ruby ///.#java (file.rooted fs "program" ruby_packager.main_file)])) / (# fs separator) cache_directory (format working_directory / target)] _ (console.write_line ..start console) diff --git a/stdlib/source/program/aedifex/command/test.lux b/stdlib/source/program/aedifex/command/test.lux index b2f67ba07..4aa1ecc54 100644 --- a/stdlib/source/program/aedifex/command/test.lux +++ b/stdlib/source/program/aedifex/command/test.lux @@ -1,28 +1,28 @@ (.using - [library - [lux "*" - [abstract - [monad {"+" do}]] - [control - [concurrency - ["[0]" async {"+" Async} ("[1]#[0]" monad)]]] - [math - [number - ["i" int]]] - [world - ["[0]" program {"+" Program}] - ["[0]" file] - ["[0]" shell {"+" Exit Shell}] - ["[0]" console {"+" Console}]]]] - ["[0]" // "_" - ["[1][0]" build] - ["/[1]" // "_" - ["[1]" profile] - ["[1][0]" action] - ["[1][0]" command {"+" Command}] - ["[1][0]" runtime] - [dependency - [resolution {"+" Resolution}]]]]) + [library + [lux "*" + [abstract + [monad {"+" do}]] + [control + [concurrency + ["[0]" async {"+" Async} ("[1]#[0]" monad)]]] + [math + [number + ["i" int]]] + [world + ["[0]" program {"+" Program}] + ["[0]" file] + ["[0]" shell {"+" Exit Shell}] + ["[0]" console {"+" Console}]]]] + ["[0]" // "_" + ["[1][0]" build] + ["/[1]" // "_" + ["[1]" profile] + ["[1][0]" action] + ["[1][0]" command {"+" Command}] + ["[1][0]" runtime] + [dependency + [resolution {"+" Resolution}]]]]) (def: .public start "[TEST STARTED]") (def: .public success "[TEST ENDED]") @@ -46,7 +46,7 @@ (|> program (///runtime.for (value@ ///.#java profile)) (//build.with_jvm_class_path host_dependencies)) - + (^template [ ] [{ artifact} (///runtime.for (value@ profile) diff --git a/stdlib/source/program/compositor.lux b/stdlib/source/program/compositor.lux index 8e0077b4a..54fc903ad 100644 --- a/stdlib/source/program/compositor.lux +++ b/stdlib/source/program/compositor.lux @@ -5,7 +5,7 @@ ["@" target] ["[0]" debug] [abstract - [monad {"+" do}]] + ["[0]" monad {"+" do}]] [control ["[0]" io {"+" IO io}] ["[0]" try {"+" Try}] @@ -87,7 +87,17 @@ (-> (file.System Async) (Dictionary file.Path Binary) [Packager file.Path] Static Archive Context (Async (Try Any))) (case (packager host_dependencies archive context) {try.#Success content} - (# fs write content package) + (case content + {.#Left content} + (# fs write content package) + + {.#Right content} + (do [! (try.with async.monad)] + [_ (# fs make_directory package) + _ (monad.each ! (function (_ [name content]) + (# fs write content (file.rooted fs package name))) + content)] + (in []))) {try.#Failure error} (# async.monad in {try.#Failure error}))) -- cgit v1.2.3