1
0
mirror of https://github.com/janet-lang/janet synced 2025-10-24 20:27:41 +00:00

Add lockfiles to jpm.

Add make-lockfile and load-lockfile commands.
This commit is contained in:
Calvin Rose
2020-03-25 19:44:30 -05:00
parent 655633ef34
commit e380c01dd1

View File

@@ -194,7 +194,8 @@
(loop [k :keys currenv :when (keyword? k)] (loop [k :keys currenv :when (keyword? k)]
(put env k (currenv k))) (put env k (currenv k)))
(dofile path :env env :exit true) (dofile path :env env :exit true)
(when-let [rules (env :rules)] (merge-into (getrules) rules))) (when-let [rules (env :rules)] (merge-into (getrules) rules))
env)
# #
# OS and shell helpers # OS and shell helpers
@@ -251,11 +252,7 @@
If we can't create it, give a friendly error. Return true if created, false if If we can't create it, give a friendly error. Return true if created, false if
existing. Throw an error if we can't create it." existing. Throw an error if we can't create it."
[dir] [dir]
(if (os/mkdir dir) (os/mkdir dir))
true
(if (os/stat dir :mode)
false
(error (string "Could not create " dir " - this could be a permission issue.")))))
# #
# C Compilation # C Compilation
@@ -607,7 +604,7 @@ int main(int argc, const char **argv) {
(defn install-git (defn install-git
"Install a bundle from git. If the bundle is already installed, the bundle "Install a bundle from git. If the bundle is already installed, the bundle
is reinistalled (but not rebuilt if artifacts are cached)." is reinistalled (but not rebuilt if artifacts are cached)."
[repotab &opt recurse] [repotab &opt recurse no-deps]
(def repo (if (string? repotab) repotab (repotab :repo))) (def repo (if (string? repotab) repotab (repotab :repo)))
(def tag (unless (string? repotab) (repotab :tag))) (def tag (unless (string? repotab) (repotab :tag)))
# prevent infinite recursion (very unlikely, but consider # prevent infinite recursion (very unlikely, but consider
@@ -639,24 +636,22 @@ int main(int argc, const char **argv) {
(rimraf module-dir) (rimraf module-dir)
(error (string "could not clone git dependency " repo)))) (error (string "could not clone git dependency " repo))))
(def olddir (os/cwd)) (def olddir (os/cwd))
(try (os/cd module-dir)
(defer (os/cd olddir)
(with-dyns [:rules @{} (with-dyns [:rules @{}
:modpath (abspath (dyn :modpath JANET_MODPATH)) :modpath (abspath (dyn :modpath JANET_MODPATH))
:headerpath (abspath (dyn :headerpath JANET_HEADERPATH)) :headerpath (abspath (dyn :headerpath JANET_HEADERPATH))
:libpath (abspath (dyn :libpath JANET_LIBPATH)) :libpath (abspath (dyn :libpath JANET_LIBPATH))
:binpath (abspath (dyn :binpath JANET_BINPATH))] :binpath (abspath (dyn :binpath JANET_BINPATH))]
(os/cd module-dir)
(unless fresh (unless fresh
(os/execute ["git" "pull" "origin" "master"] :p)) (os/execute ["git" "pull" "origin" "master"] :p))
(when tag (when tag
(os/execute ["git" "reset" "--hard" tag] :p)) (os/execute ["git" "reset" "--hard" tag] :p))
(os/execute ["git" "submodule" "update" "--init" "--recursive"] :p) (os/execute ["git" "submodule" "update" "--init" "--recursive"] :p)
(import-rules "./project.janet") (import-rules "./project.janet")
(do-rule "install-deps") (unless no-deps (do-rule "install-deps"))
(do-rule "build") (do-rule "build")
(do-rule "install")) (do-rule "install"))))
([err] (print "Error building git repository dependency: " err)))
(os/cd olddir))
(defn install-rule (defn install-rule
"Add install and uninstall rule for moving file from src into destdir." "Add install and uninstall rule for moving file from src into destdir."
@@ -669,6 +664,54 @@ int main(int argc, const char **argv) {
(mkdir destdir) (mkdir destdir)
(copy src destdir))) (copy src destdir)))
(defn- pslurp
"Like slurp, but with file/popen instead file/open. Also trims output"
[cmd]
(string/trim (with [f (file/popen cmd)] (:read f :all))))
(defn- make-lockfile
[&opt filename]
(default filename "lockfile.janet")
(def cwd (os/cwd))
(def packages @[])
(os/cd (find-cache))
(defer (os/cd cwd)
(each repo (os/dir ".")
(os/cd repo)
(def sha (pslurp "git rev-parse HEAD"))
(def url (pslurp "git remote get-url origin"))
(def deps
(with-dyns [:rules @{}]
(def env (import-rules "./project.janet"))
((env :project) :dependencies)))
(array/push packages {:repo url :sha sha :deps (or deps [])})
(os/cd "..")))
# Put in correct order, such that a package is preceded by all of its dependencies
(def ordered-packages @[])
(def resolved @{})
(while (< (length ordered-packages) (length packages))
(each p packages
(def {:repo r :sha s :deps d} p)
(unless (resolved r)
(when (all resolved d)
(array/push ordered-packages p)
(put resolved r true)))))
# Write to file
(with [f (file/open filename :w)] (with-dyns [:out f] (printf "%j" ordered-packages))))
(defn- load-lockfile
[&opt filename]
(default filename "lockfile.janet")
(def locksource (slurp filename))
(def lockarray
(let [p (parser/new)]
(:consume p locksource)
(if (= :error (:status p))
(error (string "Could not parse lockfile " filename ": " (parser/error p))))
(:produce p)))
(each {:repo url :sha sha} lockarray
(install-git {:repo url :tag sha} nil true)))
# #
# Declaring Artifacts - used in project.janet, targets specifically # Declaring Artifacts - used in project.janet, targets specifically
# tailored for janet. # tailored for janet.
@@ -815,6 +858,7 @@ int main(int argc, const char **argv) {
(print "generating " manifest "...") (print "generating " manifest "...")
(mkdir manifests) (mkdir manifests)
(spit manifest (string (string/join installed-files "\n") "\n"))) (spit manifest (string (string/join installed-files "\n") "\n")))
(phony "install" ["uninstall" "build" "manifest"] (phony "install" ["uninstall" "build" "manifest"]
(when (dyn :test) (when (dyn :test)
(do-rule "test")) (do-rule "test"))
@@ -888,6 +932,12 @@ Subcommands are:
rules : list rules available with run. rules : list rules available with run.
update-pkgs : Update the current package listing from the remote git repository selected. update-pkgs : Update the current package listing from the remote git repository selected.
quickbin entry executable : Create an executable from a janet script with a main function. quickbin entry executable : Create an executable from a janet script with a main function.
make-lockfile (lockfile) : Create a lockfile based on repositories in the cache. The
lockfile will record the exact versions of dependencies used to ensure a reproducible
build. Lockfiles are best used with applications, not libraries. The default lockfile
name is lockfile.janet.
load-lockfile (lockfile) : Install modules from a lockfile in a reproducible way. The
default lockfile name is lockfile.janet.
Keys are: Keys are:
--modpath : The directory to install modules to. Defaults to $JANET_MODPATH, $JANET_PATH, or (dyn :syspath) --modpath : The directory to install modules to. Defaults to $JANET_MODPATH, $JANET_PATH, or (dyn :syspath)
@@ -974,6 +1024,8 @@ Flags are:
"rules" list-rules "rules" list-rules
"update-pkgs" update-pkgs "update-pkgs" update-pkgs
"uninstall" uninstall-cmd "uninstall" uninstall-cmd
"make-lockfile" make-lockfile
"load-lockfile" load-lockfile
"quickbin" quickbin}) "quickbin" quickbin})
(def- args (tuple/slice (dyn :args) 1)) (def- args (tuple/slice (dyn :args) 1))