for global installations of magrathea, setting MG_ROOT can be a way to switch quickly between different repositories (esp. in combination with `cd (mg path)`). Change-Id: I4627fe78b7cc112b75ab57e7806ffd85c6d38aee Reviewed-on: https://cl.tvl.fyi/c/depot/+/4396 Tested-by: BuildkiteCI Reviewed-by: wpcarro <wpcarro@gmail.com>
		
			
				
	
	
		
			267 lines
		
	
	
	
		
			9 KiB
		
	
	
	
		
			Scheme
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
	
		
			9 KiB
		
	
	
	
		
			Scheme
		
	
	
	
	
	
| ;; magrathea helps you build planets
 | |
| ;;
 | |
| ;; it is a tiny tool designed to ease workflows in monorepos that are
 | |
| ;; modeled after the tvl depot.
 | |
| ;;
 | |
| ;; users familiar with workflows from other, larger monorepos may be
 | |
| ;; used to having a build tool that can work in any tree location.
 | |
| ;; magrathea enables this, but with nix-y monorepos.
 | |
| 
 | |
| (import (chicken base)
 | |
|         (chicken format)
 | |
|         (chicken irregex)
 | |
|         (chicken port)
 | |
|         (chicken process)
 | |
|         (chicken process-context)
 | |
|         (chicken string)
 | |
|         (matchable)
 | |
|         (only (chicken io) read-string))
 | |
| 
 | |
| (define usage #<<USAGE
 | |
| usage: mg <command> [<target>]
 | |
| 
 | |
| target:
 | |
|   a target specification with meaning inside of the repository. can
 | |
|   be absolute (starting with //) or relative to the current directory
 | |
|   (as long as said directory is inside of the repo). if no target is
 | |
|   specified, the current directory's physical target is built.
 | |
| 
 | |
|   for example:
 | |
| 
 | |
|     //tools/magrathea - absolute physical target
 | |
|     //foo/bar:baz     - absolute virtual target
 | |
|     magrathea         - relative physical target
 | |
|     :baz              - relative virtual target
 | |
| 
 | |
| commands:
 | |
|   build - build a target
 | |
|   shell - enter a shell with the target's build dependencies
 | |
|   path  - print source folder for the target
 | |
| 
 | |
| file all feedback on b.tvl.fyi
 | |
| USAGE
 | |
| )
 | |
| 
 | |
| ;; parse target definitions. trailing slashes on physical targets are
 | |
| ;; allowed for shell autocompletion.
 | |
| ;;
 | |
| ;; component ::= any string without "/" or ":"
 | |
| ;;
 | |
| ;; physical-target ::= <component>
 | |
| ;;                   | <component> "/"
 | |
| ;;                   | <component> "/" <physical-target>
 | |
| ;;
 | |
| ;; virtual-target ::= ":" <component>
 | |
| ;;
 | |
| ;; relative-target ::= <physical-target>
 | |
| ;;                   | <virtual-target>
 | |
| ;;                   | <physical-target> <virtual-target>
 | |
| ;;
 | |
| ;; root-anchor ::= "//"
 | |
| ;;
 | |
| ;; target ::= <relative-target> | <root-anchor> <relative-target>
 | |
| 
 | |
| ;; read a path component until it looks like something else is coming
 | |
| (define (read-component first port)
 | |
|   (let ((keep-reading?
 | |
|          (lambda () (not (or (eq? #\/ (peek-char port))
 | |
|                              (eq? #\: (peek-char port))
 | |
|                              (eof-object? (peek-char port)))))))
 | |
|     (let reader ((acc (list first))
 | |
|                  (condition (keep-reading?)))
 | |
|       (if condition (reader (cons (read-char port) acc) (keep-reading?))
 | |
|           (list->string (reverse acc))))))
 | |
| 
 | |
| ;; read something that started with a slash. what will it be?
 | |
| (define (read-slash port)
 | |
|   (if (eq? #\/ (peek-char port))
 | |
|       (begin (read-char port)
 | |
|              'root-anchor)
 | |
|       'path-separator))
 | |
| 
 | |
| ;; read any target token and leave port sitting at the next one
 | |
| (define (read-token port)
 | |
|   (match (read-char port)
 | |
|          [#\/ (read-slash port)]
 | |
|          [#\: 'virtual-separator]
 | |
|          [other (read-component other port)]))
 | |
| 
 | |
| ;; read a target into a list of target tokens
 | |
| (define (read-target target-str)
 | |
|   (call-with-input-string
 | |
|    target-str
 | |
|    (lambda (port)
 | |
|      (let reader ((acc '()))
 | |
|        (if (eof-object? (peek-char port))
 | |
|            (reverse acc)
 | |
|            (reader (cons (read-token port) acc)))))))
 | |
| 
 | |
| (define-record target absolute components virtual)
 | |
| (define (empty-target) (make-target #f '() #f))
 | |
| 
 | |
| (define-record-printer (target t out)
 | |
|   (fprintf out (conc (if (target-absolute t) "//" "")
 | |
|                      (string-intersperse (target-components t) "/")
 | |
|                      (if (target-virtual t) ":" "")
 | |
|                      (or (target-virtual t) ""))))
 | |
| 
 | |
| ;; parse and validate a list of target tokens
 | |
| (define parse-tokens
 | |
|   (lambda (tokens #!optional (mode 'root) (acc (empty-target)))
 | |
|     (match (cons mode tokens)
 | |
|            ;; absolute target
 | |
|            [('root . ('root-anchor . rest))
 | |
|             (begin (target-absolute-set! acc #t)
 | |
|                    (parse-tokens rest 'root acc))]
 | |
| 
 | |
|            ;; relative target minus potential garbage
 | |
|            [('root . (not ('path-separator . _)))
 | |
|             (parse-tokens tokens 'normal acc)]
 | |
| 
 | |
|            ;; virtual target
 | |
|            [('normal . ('virtual-separator . rest))
 | |
|             (parse-tokens rest 'virtual acc)]
 | |
| 
 | |
|            [('virtual . ((? string? v)))
 | |
|             (begin
 | |
|               (target-virtual-set! acc v)
 | |
|               acc)]
 | |
| 
 | |
|            ;; chomp through all components and separators
 | |
|            [('normal . ('path-separator . rest)) (parse-tokens rest 'normal acc)]
 | |
|            [('normal . ((? string? component) . rest))
 | |
|             (begin (target-components-set!
 | |
|                     acc (append (target-components acc) (list component)))
 | |
|                    (parse-tokens rest 'normal acc ))]
 | |
| 
 | |
|            ;; nothing more to parse and not in a weird state, all done, yay!
 | |
|            [('normal . ()) acc]
 | |
| 
 | |
|            ;; oh no, we ran out of input too early :(
 | |
|            [(_ . ()) `(error . ,(format "unexpected end of input while parsing ~s target" mode))]
 | |
| 
 | |
|            ;; something else was invalid :(
 | |
|            [_ `(error . ,(format "unexpected ~s while parsing ~s target" (car tokens) mode))])))
 | |
| 
 | |
| (define (parse-target target)
 | |
|   (parse-tokens (read-target target)))
 | |
| 
 | |
| ;; turn relative targets into absolute targets based on the current
 | |
| ;; directory
 | |
| (define (normalise-target t)
 | |
|   (when (not (target-absolute t))
 | |
|     (target-components-set! t (append (relative-repo-path)
 | |
|                                       (target-components t)))
 | |
|     (target-absolute-set! t #t))
 | |
|   t)
 | |
| 
 | |
| ;; nix doesn't care about the distinction between physical and virtual
 | |
| ;; targets, normalise it away
 | |
| (define (normalised-components t)
 | |
|   (if (target-virtual t)
 | |
|       (append (target-components t) (list (target-virtual t)))
 | |
|       (target-components t)))
 | |
| 
 | |
| ;; return the current repository root as a string
 | |
| (define mg--repository-root #f)
 | |
| (define (repository-root)
 | |
|   (or mg--repository-root
 | |
|       (begin
 | |
|         (set! mg--repository-root
 | |
|               (or (get-environment-variable "MG_ROOT")
 | |
|                   (string-chomp
 | |
|                    (call-with-input-pipe "git rev-parse --show-toplevel"
 | |
|                                          (lambda (p) (read-string #f p))))))
 | |
|         mg--repository-root)))
 | |
| 
 | |
| ;; determine the current path relative to the root of the repository
 | |
| ;; and return it as a list of path components.
 | |
| (define (relative-repo-path)
 | |
|   (string-split
 | |
|    (substring (current-directory) (string-length (repository-root))) "/"))
 | |
| 
 | |
| ;; escape a string for interpolation in nix code
 | |
| (define (nix-escape str)
 | |
|   (string-translate* str '(("\"" . "\\\"")
 | |
|                            ("${" . "\\${"))))
 | |
| 
 | |
| ;; create a nix expression to build the attribute at the specified
 | |
| ;; components
 | |
| ;;
 | |
| ;; an empty target will build the current folder instead.
 | |
| ;;
 | |
| ;; this uses builtins.getAttr explicitly to avoid problems with
 | |
| ;; escaping.
 | |
| (define (nix-expr-for target)
 | |
|   (let nest ((parts (normalised-components (normalise-target target)))
 | |
|              (acc (conc "(import " (repository-root) " {})")))
 | |
|     (match parts
 | |
|            [() (conc "with builtins; " acc)]
 | |
|            [_ (nest (cdr parts)
 | |
|                     (conc "(getAttr \""
 | |
|                           (nix-escape (car parts))
 | |
|                           "\" " acc ")"))])))
 | |
| 
 | |
| ;; exit and complain at the user if something went wrong
 | |
| (define (mg-error message)
 | |
|   (format (current-error-port) "[mg] error: ~A~%" message)
 | |
|   (exit 1))
 | |
| 
 | |
| (define (guarantee-success value)
 | |
|   (match value
 | |
|          [('error . message) (mg-error message)]
 | |
|          [_ value]))
 | |
| 
 | |
| (define (execute-build t)
 | |
|   (let ((expr (nix-expr-for t)))
 | |
|     (printf "[mg] building target ~A~%" t)
 | |
|     (process-execute "nix-build" (list "-E" expr "--show-trace"))))
 | |
| 
 | |
| (define (build args)
 | |
|   (match args
 | |
|          ;; simplest case: plain mg build with no target spec -> build
 | |
|          ;; the current folder's main target.
 | |
|          [() (execute-build (empty-target))]
 | |
| 
 | |
|          ;; single argument should be a target spec
 | |
|          [(arg) (execute-build
 | |
|                  (guarantee-success (parse-target arg)))]
 | |
| 
 | |
|          [other (print "not yet implemented")]))
 | |
| 
 | |
| (define (execute-shell t)
 | |
|   (let ((expr (nix-expr-for t))
 | |
|         (user-shell (or (get-environment-variable "SHELL") "bash")))
 | |
|     (printf "[mg] entering shell for ~A~%" t)
 | |
|     (process-execute "nix-shell"
 | |
|                      (list "-E" expr "--command" user-shell))))
 | |
| 
 | |
| (define (shell args)
 | |
|   (match args
 | |
|          [() (execute-shell (empty-target))]
 | |
|          [(arg) (execute-shell
 | |
|                  (guarantee-success (parse-target arg)))]
 | |
|          [other (print "not yet implemented")]))
 | |
| 
 | |
| (define (path args)
 | |
|   (match args
 | |
|          [(arg)
 | |
|           (print (conc (repository-root)
 | |
|                        "/"
 | |
|                        (string-intersperse
 | |
|                         (target-components
 | |
|                          (normalise-target
 | |
|                           (guarantee-success (parse-target arg)))) "/")))]
 | |
|          [() (mg-error "path command needs a target")]
 | |
|          [other (mg-error (format "unknown arguments: ~a" other))]))
 | |
| 
 | |
| (define (main args)
 | |
|   (match args
 | |
|          [() (print usage)]
 | |
|          [("build" . _) (build (cdr args))]
 | |
|          [("shell" . _) (shell (cdr args))]
 | |
|          [("path" . _) (path (cdr args))]
 | |
|          [other (begin (print "unknown command: mg " args)
 | |
|                        (print usage))]))
 | |
| 
 | |
| (main (command-line-arguments))
 |