phc-toolkit/ids.rkt
2017-04-27 23:38:55 +02:00

321 lines
12 KiB
Racket

#lang typed/racket
(require "typed-untyped.rkt")
(define-typed/untyped-modules #:no-test #:untyped-first
(provide !temp
(rename-out [!temp &])
format-ids
hyphen-ids
format-temp-ids
#|!temp|#
define-temp-ids)
(require "typed-untyped.rkt"
"untyped-only/syntax-parse.rkt")
(require-typed/untyped "sequence.rkt")
(if-typed (require phc-toolkit/aliases)
(require phc-toolkit/untyped/aliases))
(begin-for-syntax (require "typed-untyped.rkt"
"untyped-only/format-id-record.rkt")
(if-typed (require phc-toolkit/aliases)
(require phc-toolkit/untyped/aliases)))
(module m-!temp racket
(provide !temp)
(require syntax/parse
syntax/parse/experimental/template)
(define-template-metafunction (!temp stx)
(syntax-parse stx
[(_ id:id)
#:with (temp) (generate-temporaries #'(id))
#'temp]
#|[(_ . id:id)
#:with (temp) (generate-temporaries #'(id))
#'temp]
[(_ id:id ...)
(generate-temporaries #'(id ...))]|#)))
(require 'm-!temp)
(require/typed racket/syntax
[format-id ( Syntax String (U String Identifier) *
Identifier)])
(require (only-in racket/syntax define/with-syntax)
(only-in syntax/stx stx-map)
(for-syntax racket/base
racket/format
racket/syntax
syntax/parse
syntax/parse/experimental/template))
;(require racket/sequence) ;; in-syntax
(define-type S-Id-List
(U String
Identifier
(Listof String)
(Listof Identifier)
(Syntaxof (Listof Identifier))))
; TODO: format-ids doesn't accept arbitrary values. Should we change it?
;
(: format-ids ( (U Syntax ( (U String Identifier) * Syntax))
String
S-Id-List *
(Listof Identifier)))
(define (format-ids lex-ctx format . vs)
(let* ([seqs
(map (λ ([v : S-Id-List])
(cond
[(string? v) (in-cycle (in-value v))]
[(identifier? v) (in-cycle (in-value v))]
[(list? v) (in-list v)]
[else (in-list (syntax->list v))]))
vs)]
[justconstants (andmap (λ (x) (or (string? x) (identifier? x))) vs)]
[seqlst (apply sequence-list seqs)])
(for/list : (Listof Identifier)
([items seqlst]
[bound-length (if justconstants
(in-value 'yes)
(in-cycle (in-value 'no)))])
(apply format-id
(if (procedure? lex-ctx) (apply lex-ctx items) lex-ctx)
format
items))))
(: hyphen-ids ( (U Syntax ( (U String Identifier) * Syntax))
S-Id-List *
(Listof Identifier)))
(define (hyphen-ids lex-ctx . vs)
(apply format-ids
lex-ctx
(string-join (map (λ _ "~a") vs) "-")
vs))
(: format-temp-ids ( String
S-Id-List *
(Listof Identifier)))
(define (format-temp-ids format . vs)
;; Introduce the binding in a fresh scope.
(apply format-ids
(λ _ ((make-syntax-introducer) (if (syntax? format)
format
(datum->syntax #f '()))))
format
vs))
(: to-identifier ( Any Identifier))
(define (to-identifier v)
(cond
[(identifier? v) v]
[(syntax? v) (datum->syntax v (to-symbol (syntax-e v)))]
[else (datum->syntax #f (to-symbol v))]))
(: to-symbol ( Any Symbol))
(define (to-symbol v)
(cond
[(symbol? v) v]
[(string? v) (string->symbol v)]
[(number? v) (string->symbol (format "~a" v))]
[else (syntax-e (generate-temporary v))]))
(: generate-string ( String))
(define (generate-string)
(symbol->string
(syntax-e
(generate-temporary ""))))
(require (for-syntax (submod "stx.rkt" untyped)))
(: curried-map-on-attribute-step
( (A B) ( ( A B)
(case→ ( #f #f)
( (Listof A) (Listof B))
( (U #f (Listof A))
(U #f (Listof B)))))))
(define ((curried-map-on-attribute-step f) l)
(if (eq? l #f)
l
(map f l)))
(: curried-map-on-attribute-last
( (A B) ( ( (Syntaxof A) B)
(case→ ( #f #f)
( (Syntaxof A) B)
( (U #f (Syntaxof A)) (U #f B))))))
(define ((curried-map-on-attribute-last f) v)
(if (eq? v #f)
v
(f v)))
;; (map-on-attribute f depth)
(define-syntax (map-on-attribute stx)
(syntax-case stx ()
[(_ f 0)
#'(curried-map-on-attribute-last f)]
[(_ f depth)
#`(curried-map-on-attribute-step
(map-on-attribute f
#,(sub1 (syntax-e #'depth))))]))
(begin-for-syntax
(define-syntax-class dotted
(pattern id:id
#:attr make-dotted
(λ (x) x)
#:attr wrap
(λ (x f) (f x #t))
#:attr depth 0
#:with stx-depth #'0)
(pattern (nested:dotted (~literal ...));(~and dots (~literal ...)) ...+)
#:with id #'nested.id
#:attr make-dotted
(λ (x) #`(#,((attribute nested.make-dotted) x) (... ...)))
#:attr wrap
(λ (x f) (f ((attribute nested.wrap) x f) #f))
#:attr depth (add1 (attribute nested.depth))
#:with stx-depth #`#,(add1 (attribute nested.depth))))
(define-syntax-class simple-format
(pattern format
#:when (string? (syntax-e #'format))
#:when (regexp-match #rx"^([^~]|~~)*~a([^~]|~~)*$"
(syntax-e #'format)))))
;; This macro should really be heavily refactored.
;; TODO: merge all cases thanks to format-id/record and syntax classes.
(define-syntax (define-temp-ids stx)
(with-arrows
(syntax-parse stx
#|
;; TODO : factor this with the next case.
[(_ format ((base:id (~literal ...)) (~literal ...)))
#:when (string? (syntax-e #'format))
(with-syntax ([pat (format-id #'format (syntax-e #'format) #'base)])
#'(define/with-syntax ((pat (... ...)) (... ...))
(stx-map (curry format-temp-ids format)
#'((base (... ...)) (... ...)))))]
|#
;; Multiple formats
[(_ {~and {~optional #:concise} {~seq maybe-concise }}
(format:simple-format )
(~and (~seq options )
(~seq base:dotted
(~or (~seq #:first-base first-base)
(~optional (~seq #:first first)))
(~optional (~seq #:prefix prefix)))))
#'(begin (define-temp-ids maybe-concise format options ) )]
;; New features (arrows and #:first) special-cased for now
;; TODO: make these features more general.
;; With #:first-base, translated to #:first
[(_ {~and {~optional #:concise} {~seq maybe-concise }}
format:simple-format base:dotted
#:first-base first-base
(~optional (~seq #:prefix prefix)))
#:with first (format-id/record #'format #'format #'first-base)
(template
(define-temp-ids maybe-concise format base
#:first first
(?? (?@ #:prefix prefix))))]
;; Base case with a simple format "...~a...".
[(_ {~optional {~and #:concise concise?}}
format:simple-format
base:dotted
(~optional (~seq #:first first))
(~optional (~seq #:first… first…))
(~optional (~seq #:prefix prefix)))
(let* ([base-len (string-length (symbol->string (syntax-e #'base.id)))])
(define/with-syntax pat
(format-id/record #'format #'format #'base.id))
(define/with-syntax pat-dotted ((attribute base.make-dotted) #'pat))
(define/with-syntax maybe-generate-temporary
(if (attribute concise?)
#'to-identifier
#'generate-temporary))
(define/with-syntax format-temp-ids-last
(template
(λ (x)
(car (format-temp-ids (?? (?@ (string-append "~a:" format) prefix)
format)
(maybe-generate-temporary x))))))
(define/with-syntax format-temp-ids*
#'(map-on-attribute format-temp-ids-last base.stx-depth))
(define/with-syntax (tmp-valvar) (generate-temporaries #`(base.id)))
(define/with-syntax do-define-pat
(syntax-parse (attribute-info #'base.id '(pvar attr))
[({~datum attr} valvar depth name syntax?)
#'(define-raw-attribute pat
tmp-valvar
(format-temp-ids* valvar)
depth
syntax?)]
[({~datum pvar} valvar depth)
#'(define-raw-syntax-mapping pat
tmp-valvar
(format-temp-ids* valvar)
depth)]))
(define/with-syntax do-define-first…
(if (attribute first…)
(let ()
(define/with-syntax (tmp-first-valvar)
(generate-temporaries #`(base.id)))
(syntax-parse (attribute-info #'base.id '(pvar attr))
[({~datum attr} valvar depth name syntax?)
;; TODO: always define an attribute, but don't use
;; define-raw-attribute, instead use the copy-attribute
;; code from subtemplate.
#`(define-raw-attribute first…
tmp-first-valvar
(car tmp-valvar)
#,(sub1 (syntax-e #'depth))
syntax?)]
[({~datum pvar} valvar depth)
#`(define-raw-syntax-mapping first…
tmp-first-valvar
(car tmp-valvar)
#,(sub1 (syntax-e #'depth)))]))
#'(begin)))
(define/with-syntax do-define-first
(if (attribute first)
#'(define/with-syntax (first . _)
#'pat-dotted)
#'(begin)))
#'(begin do-define-pat
do-define-first
do-define-first…))]
;; Simplistic handling when the format contains no ~ at all.
;; (TODO: should allow ~~)
[(_ {~optional {~and #:concise concise?}} format base:dotted)
#:when (string? (syntax-e #'format))
#:when (regexp-match #rx"^([^~]|~~)*$" (syntax-e #'format))
(define/with-syntax pat (format-id/record #'format #'format))
(define/with-syntax pat-dotted ((attribute base.make-dotted) #'pat))
(define/with-syntax format-temp-ids*
((attribute base.wrap) #`(λ (x)
#,(if (attribute concise?)
#'(car (format-temp-ids
(string-append format)))
#'(car (format-temp-ids
(string-append format "-~a")
(generate-string)))))
(λ (x deepest?)
(if deepest?
x
#`(curry stx-map #,x)))))
#'(define/with-syntax pat-dotted
(format-temp-ids* #'base))]
;; Very simplistic handling when the name is explicitly given.
[(_ {~optional {~and #:concise concise?}}
name:id format:expr . vs)
#`(define/with-syntax name (format-temp-ids format . vs))]))))