2020-02-19 22:49:50 +01:00
|
|
|
#lang racket
|
2020-02-16 21:39:42 +01:00
|
|
|
|
|
|
|
;;; dds/utils
|
|
|
|
|
|
|
|
;;; Various utilities.
|
|
|
|
|
2020-02-22 21:00:34 +01:00
|
|
|
(require
|
|
|
|
graph
|
|
|
|
(for-syntax syntax/parse racket/list))
|
2020-02-16 21:39:42 +01:00
|
|
|
|
2020-02-19 23:37:06 +01:00
|
|
|
(provide
|
2020-02-20 14:19:30 +01:00
|
|
|
;; Functions
|
2020-02-19 23:37:06 +01:00
|
|
|
(contract-out [eval-with (-> variable-mapping? any/c any)]
|
2020-02-20 14:19:30 +01:00
|
|
|
[extract-symbols (-> any/c list?)]
|
2020-02-22 10:29:34 +01:00
|
|
|
[any->string (-> any/c string?)]
|
2020-02-22 12:12:57 +01:00
|
|
|
[stringify-variable-mapping (-> variable-mapping? string-variable-mapping?)]
|
2020-02-22 12:18:37 +01:00
|
|
|
[string->any (-> string? any/c)]
|
2020-02-22 12:27:28 +01:00
|
|
|
[read-org-table (-> string? (listof any/c))]
|
2020-02-26 15:51:00 +01:00
|
|
|
[unstringify-pairs (-> (listof (general-pair/c string? any/c))
|
2020-02-22 19:15:39 +01:00
|
|
|
(listof (general-pair/c symbol? any/c)))]
|
2020-02-23 19:17:16 +01:00
|
|
|
[read-org-variable-mapping (-> string? variable-mapping?)]
|
|
|
|
[update-vertices/unweighted (-> graph? (-> any/c any/c) graph?)])
|
2020-02-22 10:36:03 +01:00
|
|
|
;; Contracts
|
|
|
|
(contract-out [variable-mapping? contract?]
|
2020-02-22 19:14:46 +01:00
|
|
|
[string-variable-mapping? contract?]
|
|
|
|
[general-pair/c (-> contract? contract? contract?)])
|
2020-02-19 23:37:06 +01:00
|
|
|
;; Syntax
|
2020-02-22 21:00:34 +01:00
|
|
|
auto-hash-ref/explicit auto-hash-ref/: sgfy unorg dotit)
|
2020-02-16 21:39:42 +01:00
|
|
|
|
2020-02-18 00:00:18 +01:00
|
|
|
;;; ===================
|
2020-02-16 21:39:42 +01:00
|
|
|
;;; HashTable Injection
|
2020-02-18 00:00:18 +01:00
|
|
|
;;; ===================
|
2020-02-16 21:39:42 +01:00
|
|
|
|
|
|
|
;;; This section of the file contains some utilities to streamline the
|
|
|
|
;;; usage of hash tables mapping symbols to values. The goal is
|
|
|
|
;;; essentially to avoid having to write explicit hash-ref calls.
|
|
|
|
|
2020-02-20 00:21:10 +01:00
|
|
|
;;; A variable mapping is a hash table mapping symbols to values.
|
2020-02-22 10:35:01 +01:00
|
|
|
(define (variable-mapping? dict) (hash/c symbol? any/c))
|
2020-02-19 23:25:00 +01:00
|
|
|
|
2020-02-16 21:39:42 +01:00
|
|
|
;;; Given a (HashTable Symbol a) and a sequence of symbols, binds
|
|
|
|
;;; these symbols to the values they are associated to in the hash
|
|
|
|
;;; table, then puts the body in the context of these bindings.
|
2020-02-17 00:16:01 +01:00
|
|
|
;;;
|
|
|
|
;;; > (let ([ht #hash((a . 1) (b . 2))])
|
|
|
|
;;; (auto-hash-ref/explicit (ht a b) (+ a (* 2 b))))
|
|
|
|
;;; 5
|
|
|
|
;;;
|
2020-02-17 00:27:04 +01:00
|
|
|
;;; Note that only one expression can be supplied in the body.
|
2020-02-16 21:39:42 +01:00
|
|
|
(define-syntax (auto-hash-ref/explicit stx)
|
|
|
|
(syntax-parse stx
|
|
|
|
[(_ (ht:id xs:id ...) body:expr)
|
|
|
|
#`(let #,(for/list ([x (syntax->list #'(xs ...))])
|
|
|
|
#`[#,x (hash-ref ht '#,x)])
|
|
|
|
body)]))
|
|
|
|
|
2020-02-17 00:16:44 +01:00
|
|
|
;;; Given an expression and a (HashTable Symbol a), looks up the
|
|
|
|
;;; symbols with a leading semicolon and binds them to the value they
|
|
|
|
;;; are associated to in the hash table.
|
|
|
|
;;;
|
|
|
|
;;; > (let ([ht #hash((a . 1) (b . 2))])
|
|
|
|
;;; (auto-hash-ref/: ht (+ :a (* 2 :b))))
|
|
|
|
;;; 5
|
|
|
|
;;;
|
|
|
|
;;; Note that the symbol :a is matched to the key 'a in the hash
|
|
|
|
;;; table.
|
2020-02-17 00:27:04 +01:00
|
|
|
;;;
|
|
|
|
;;; Note that only one expression can be supplied in the body.
|
2020-02-17 00:16:44 +01:00
|
|
|
(define-syntax (auto-hash-ref/: stx)
|
|
|
|
(syntax-parse stx
|
|
|
|
[(_ ht:id body)
|
|
|
|
(let* ([names/: (collect-colons (syntax->datum #'body))])
|
|
|
|
#`(let #,(for/list ([x names/:])
|
|
|
|
;; put x in the same context as body
|
|
|
|
#`[#,(datum->syntax #'body x)
|
|
|
|
(hash-ref ht '#,(strip-colon x))])
|
|
|
|
body))]))
|
|
|
|
|
|
|
|
;;; The helper functions for auto-hash-ref/:.
|
|
|
|
(begin-for-syntax
|
|
|
|
;; Collect all the symbols starting with a colon in datum.
|
|
|
|
(define (collect-colons datum)
|
|
|
|
(remove-duplicates
|
|
|
|
(flatten
|
|
|
|
(for/list ([token datum])
|
|
|
|
(cond
|
|
|
|
[(symbol? token)
|
|
|
|
(let ([name (symbol->string token)])
|
|
|
|
(if (eq? #\: (string-ref name 0))
|
|
|
|
token
|
|
|
|
'()))]
|
|
|
|
[(list? token)
|
|
|
|
(collect-colons token)]
|
|
|
|
[else '()])))))
|
2020-02-16 21:39:42 +01:00
|
|
|
|
2020-02-17 00:16:44 +01:00
|
|
|
;; Strip the leading colon off x.
|
|
|
|
(define (strip-colon x)
|
|
|
|
(let ([x-str (symbol->string x)])
|
|
|
|
(if (eq? #\: (string-ref x-str 0))
|
|
|
|
(string->symbol (substring x-str 1))
|
|
|
|
x))))
|
2020-02-17 23:52:15 +01:00
|
|
|
|
|
|
|
;;; Temporarily injects the mappings from the given hash table as
|
2020-02-19 22:51:49 +01:00
|
|
|
;;; bindings in a namespace including racket/base and then evaluates
|
|
|
|
;;; the expression.
|
2020-02-17 23:52:15 +01:00
|
|
|
;;;
|
|
|
|
;;; > (let ([ht #hash((a . 1) (b . 1))])
|
|
|
|
;;; (eval-with ht '(+ b a 1)))
|
|
|
|
;;; 3
|
|
|
|
;;;
|
|
|
|
;;; The local bindings from the current lexical scope are not
|
|
|
|
;;; conserved. Therefore, the following outputs an error about a
|
|
|
|
;;; missing identifier:
|
|
|
|
;;;
|
|
|
|
;;; > (let ([ht #hash((a . 1) (b . 1))]
|
|
|
|
;;; [z 1])
|
|
|
|
;;; (eval-with ht '(+ b z a 1)))
|
|
|
|
;;;
|
|
|
|
(define (eval-with ht expr)
|
2020-02-19 22:49:50 +01:00
|
|
|
(parameterize ([current-namespace (make-base-namespace)])
|
2020-02-23 08:57:50 +01:00
|
|
|
(for ([(x val) ht]) (namespace-set-variable-value! x val))
|
2020-02-17 23:52:15 +01:00
|
|
|
(eval expr)))
|
|
|
|
|
|
|
|
;;; Same as eval-with, but returns only the first value produced by
|
|
|
|
;;; the evaluated expression.
|
|
|
|
(define (eval-with1 ht expr)
|
|
|
|
(let ([vals (call-with-values (λ () (eval-with ht expr))
|
|
|
|
(λ vals vals))])
|
|
|
|
(car vals)))
|
2020-02-19 22:11:44 +01:00
|
|
|
|
|
|
|
|
|
|
|
;;; ==============================
|
|
|
|
;;; Analysis of quoted expressions
|
|
|
|
;;; ==============================
|
|
|
|
|
|
|
|
;;; Produces a list of symbols appearing in the quoted expression
|
|
|
|
;;; passed in the first argument.
|
|
|
|
(define (extract-symbols form)
|
2020-02-23 09:00:54 +01:00
|
|
|
(match form
|
|
|
|
[(? symbol?) (list form)]
|
|
|
|
[(? list?) (flatten (for/list ([x form])
|
|
|
|
(extract-symbols x)))]
|
2020-02-19 22:11:44 +01:00
|
|
|
[else '()]))
|
2020-02-21 18:01:08 +01:00
|
|
|
|
|
|
|
|
2020-02-22 20:51:32 +01:00
|
|
|
;;; =========================
|
|
|
|
;;; Interaction with Org-mode
|
|
|
|
;;; =========================
|
2020-02-21 18:01:08 +01:00
|
|
|
|
|
|
|
;;; Org-mode supports laying out the output of code blocks as tables,
|
|
|
|
;;; which is very practical for various variable mappings (e.g.,
|
|
|
|
;;; states). However, when the hash table maps variables to lists,
|
|
|
|
;;; Org-mode will create a column per list element, which may or may
|
|
|
|
;;; not be the desired effect. In this section I define some
|
2020-02-22 20:51:32 +01:00
|
|
|
;;; utilities for nicer interoperation with Org-mode tables. I also
|
|
|
|
;;; define some shortcuts to reduce the number of words to type when
|
|
|
|
;;; using dds with Org-mode. See example.org for examples of usage.
|
2020-02-21 18:01:08 +01:00
|
|
|
|
|
|
|
;;; Converts any value to string.
|
|
|
|
(define (any->string x)
|
|
|
|
(with-output-to-string (λ () (display x))))
|
2020-02-22 10:29:34 +01:00
|
|
|
|
|
|
|
;;; A string variable mapping is a mapping from variables to strings.
|
2020-02-22 10:35:01 +01:00
|
|
|
(define (string-variable-mapping? dict) (hash/c symbol? string?))
|
2020-02-22 10:29:34 +01:00
|
|
|
|
|
|
|
;;; Converts all the values of a variable mapping to string.
|
2020-02-22 12:12:57 +01:00
|
|
|
(define (stringify-variable-mapping ht)
|
2020-02-23 08:57:50 +01:00
|
|
|
(for/hash ([(key val) ht]) (values key (any->string val))))
|
2020-02-22 10:40:40 +01:00
|
|
|
|
|
|
|
;;; A shortcut for variable-mapping-stingify.
|
2020-02-22 12:12:57 +01:00
|
|
|
(define-syntax-rule (sgfy ht) (stringify-variable-mapping ht))
|
2020-02-22 12:11:37 +01:00
|
|
|
|
|
|
|
;;; Reads any value from string.
|
|
|
|
(define (string->any str)
|
|
|
|
(with-input-from-string str (λ () (read))))
|
2020-02-22 12:18:37 +01:00
|
|
|
|
|
|
|
;;; Reads a table from a string produced by Org-mode for a named
|
|
|
|
;;; table. See example.org for examples.
|
|
|
|
(define (read-org-table str) (string->any str))
|
2020-02-22 12:27:28 +01:00
|
|
|
|
2020-02-22 19:14:46 +01:00
|
|
|
;;; A contract allowing pairs constructed via cons or via list.
|
|
|
|
(define (general-pair/c key-contract val-contract)
|
|
|
|
(or/c (list/c key-contract val-contract)
|
|
|
|
(cons/c key-contract val-contract)))
|
|
|
|
|
2020-02-26 15:51:00 +01:00
|
|
|
;;; Given a list of pairs of strings and some other values (possibly
|
|
|
|
;;; strings), converts the first element of each pair to a string, and
|
|
|
|
;;; reads the second element with string->any or keeps it as is if it
|
|
|
|
;;; is not a string.
|
2020-02-22 12:27:28 +01:00
|
|
|
(define (unstringify-pairs pairs)
|
|
|
|
(for/list ([p pairs])
|
2020-02-22 18:58:19 +01:00
|
|
|
(match p
|
|
|
|
[(list key val)
|
2020-02-26 15:51:00 +01:00
|
|
|
(cons (string->symbol key) (if (string? val)
|
|
|
|
(string->any val)
|
|
|
|
val))]
|
2020-02-22 18:58:19 +01:00
|
|
|
[(cons key val) ; also handle improper pairs
|
2020-02-26 15:51:00 +01:00
|
|
|
(cons (string->symbol key) (if (string? val)
|
|
|
|
(string->any val)
|
|
|
|
val))])))
|
2020-02-22 19:15:39 +01:00
|
|
|
|
|
|
|
;;; Reads a variable mapping from a string, such as the one which
|
|
|
|
;;; Org-mode produces from tables.
|
|
|
|
(define (read-org-variable-mapping str)
|
2020-02-22 19:20:45 +01:00
|
|
|
(make-hash (unstringify-pairs (read-org-table str))))
|
2020-02-22 19:48:39 +01:00
|
|
|
|
|
|
|
;;; A synonym for read-org-variable-mapping.
|
|
|
|
(define-syntax-rule (unorg str) (read-org-variable-mapping str))
|
2020-02-22 21:00:34 +01:00
|
|
|
|
|
|
|
;;; Typeset the graph via graphviz and display it.
|
|
|
|
(define-syntax-rule (dotit gr) (display (graphviz gr)))
|
2020-02-23 19:17:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
;;; ==========================
|
|
|
|
;;; Additional graph utilities
|
|
|
|
;;; ==========================
|
|
|
|
|
|
|
|
;;; Apply a transformation to every vertex in the unweighted graph,
|
|
|
|
;;; return the new graph. If the transformation function maps two
|
|
|
|
;;; vertices to the same values, these vertices will be merged in the
|
|
|
|
;;; resulting graph. The transformation function may be called
|
|
|
|
;;; multiple times for the same vertex.
|
|
|
|
;;;
|
|
|
|
;;; This function does not rely on rename-vertex!, so it can be used
|
|
|
|
;;; to permute vertex labels.
|
|
|
|
(define (update-vertices/unweighted gr func)
|
|
|
|
(unweighted-graph/directed
|
|
|
|
(for/list ([e (in-edges gr)])
|
|
|
|
(match-let ([(list u v) e])
|
|
|
|
(list (func u) (func v))))))
|