** update inline comments
This commit is contained in:
parent
7e30256efc
commit
03899c77f1
|
@ -1,4 +1,4 @@
|
||||||
glob
|
trivial
|
||||||
Copyright (c) 2015 Ben Greenman
|
Copyright (c) 2015 Ben Greenman
|
||||||
|
|
||||||
This package is distributed under the GNU Lesser General Public
|
This package is distributed under the GNU Lesser General Public
|
||||||
|
|
21
format.rkt
21
format.rkt
|
@ -1,5 +1,7 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Statically-checked format strings
|
||||||
|
|
||||||
(provide
|
(provide
|
||||||
format:
|
format:
|
||||||
;; (-> (x:String) Any *N Void)
|
;; (-> (x:String) Any *N Void)
|
||||||
|
@ -17,7 +19,10 @@
|
||||||
)
|
)
|
||||||
|
|
||||||
(require
|
(require
|
||||||
(for-syntax typed/racket/base syntax/parse racket/sequence))
|
(for-syntax
|
||||||
|
typed/racket/base
|
||||||
|
syntax/parse
|
||||||
|
racket/sequence))
|
||||||
|
|
||||||
;; =============================================================================
|
;; =============================================================================
|
||||||
|
|
||||||
|
@ -29,7 +34,7 @@
|
||||||
[num-expected (length type*)]
|
[num-expected (length type*)]
|
||||||
[num-given (for/sum ([a (in-syntax #'(arg* ...))]) 1)])
|
[num-given (for/sum ([a (in-syntax #'(arg* ...))]) 1)])
|
||||||
(unless (= num-expected num-given)
|
(unless (= num-expected num-given)
|
||||||
(raise-arity-error
|
(apply raise-arity-error
|
||||||
(syntax-e #'f)
|
(syntax-e #'f)
|
||||||
num-expected
|
num-expected
|
||||||
(for/list ([a (in-syntax #'(arg* ...))]) (syntax->datum a))))
|
(for/list ([a (in-syntax #'(arg* ...))]) (syntax->datum a))))
|
||||||
|
@ -45,6 +50,7 @@
|
||||||
[(f tmp arg* ...)
|
[(f tmp arg* ...)
|
||||||
(syntax/loc #'f (format tmp arg* ...))]))
|
(syntax/loc #'f (format tmp arg* ...))]))
|
||||||
|
|
||||||
|
;; Short for `(displayln (format: ...))`
|
||||||
(define-syntax printf:
|
(define-syntax printf:
|
||||||
(syntax-parser
|
(syntax-parser
|
||||||
[f:id
|
[f:id
|
||||||
|
@ -55,10 +61,10 @@
|
||||||
;; -----------------------------------------------------------------------------
|
;; -----------------------------------------------------------------------------
|
||||||
|
|
||||||
;; Count the number of format escapes in a string.
|
;; Count the number of format escapes in a string.
|
||||||
;; Returns a list of optional types (to be spliced into the source code)
|
;; Returns a list of optional types (to be spliced into the source code).
|
||||||
;; Example: If result is '(#f Integer), then
|
;; Example: If result is '(#f Integer), then
|
||||||
;; - Expect 2 arguments to format string
|
;; - The format string expects 2 arguments
|
||||||
;; - First argument has no constraints, second must be an Integer
|
;; - First argument has no type constraints, second must be an Integer
|
||||||
;; (: count-format-escapes (->* [String] [#:src (U #f Syntax)] (Listof (U #f Syntax))))
|
;; (: count-format-escapes (->* [String] [#:src (U #f Syntax)] (Listof (U #f Syntax))))
|
||||||
(define-for-syntax (template->type* str #:src [stx #f])
|
(define-for-syntax (template->type* str #:src [stx #f])
|
||||||
(define last-index (- (string-length str) 1))
|
(define last-index (- (string-length str) 1))
|
||||||
|
@ -67,7 +73,7 @@
|
||||||
[(>= i last-index)
|
[(>= i last-index)
|
||||||
(reverse acc)]
|
(reverse acc)]
|
||||||
[(eq? #\~ (string-ref str i))
|
[(eq? #\~ (string-ref str i))
|
||||||
;; From fprintf docs
|
;; From fprintf docs @ http://docs.racket-lang.org/reference/Writing.html
|
||||||
(case (string-ref str (+ i 1))
|
(case (string-ref str (+ i 1))
|
||||||
[(#\% #\n #\~ #\space #\tab #\newline)
|
[(#\% #\n #\~ #\space #\tab #\newline)
|
||||||
;; Need 0 arguments
|
;; Need 0 arguments
|
||||||
|
@ -93,4 +99,3 @@
|
||||||
(string-ref str (+ i 1)))])]
|
(string-ref str (+ i 1)))])]
|
||||||
[else
|
[else
|
||||||
(loop (+ i 1) acc)])))
|
(loop (+ i 1) acc)])))
|
||||||
|
|
||||||
|
|
2
info.rkt
2
info.rkt
|
@ -4,6 +4,6 @@
|
||||||
(define deps '("base"))
|
(define deps '("base"))
|
||||||
(define build-deps '("scribble-lib" "racket-doc" "rackunit-lib"))
|
(define build-deps '("scribble-lib" "racket-doc" "rackunit-lib"))
|
||||||
(define pkg-desc "Strongly-typed macros")
|
(define pkg-desc "Strongly-typed macros")
|
||||||
(define version "0.0")
|
(define version "0.1")
|
||||||
(define pkg-authors '(ben))
|
(define pkg-authors '(ben))
|
||||||
(define scribblings '(("scribblings/trivial.scrbl")))
|
(define scribblings '(("scribblings/trivial.scrbl")))
|
||||||
|
|
20
math.rkt
20
math.rkt
|
@ -1,14 +1,18 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Constant-folding math operators.
|
||||||
|
;; Where possible, they simplify their arguments.
|
||||||
|
|
||||||
(provide
|
(provide
|
||||||
+: -: *: /:
|
+: -: *: /:
|
||||||
;; Fold syntactic constants
|
;; Same signature as the racket/base operators,
|
||||||
|
;; but try to simplify arguments during expansion.
|
||||||
)
|
)
|
||||||
|
|
||||||
(require (for-syntax
|
(require (for-syntax
|
||||||
racket/base
|
typed/racket/base
|
||||||
(only-in racket/format ~a)
|
(only-in racket/format ~a)
|
||||||
racket/syntax
|
(only-in racket/syntax format-id)
|
||||||
syntax/id-table
|
syntax/id-table
|
||||||
syntax/parse
|
syntax/parse
|
||||||
trivial/private/common
|
trivial/private/common
|
||||||
|
@ -41,10 +45,14 @@
|
||||||
|
|
||||||
;; -----------------------------------------------------------------------------
|
;; -----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
;; Simplify a list of expressions using an associative binary operator.
|
||||||
|
;; Return either:
|
||||||
|
;; - A numeric value
|
||||||
|
;; - A list of syntax objects, to be spliced back in the source code
|
||||||
(define-for-syntax (reduce/op op e*)
|
(define-for-syntax (reduce/op op e*)
|
||||||
(let loop ([prev #f]
|
(let loop ([prev #f] ;; (U #f Number), candidate for reduction
|
||||||
[acc '()]
|
[acc '()] ;; (Listof Syntax), irreducible arguments
|
||||||
[e* e*])
|
[e* e*]) ;; (Listof Syntax), arguments to process
|
||||||
(if (null? e*)
|
(if (null? e*)
|
||||||
;; then: finished, return a number (prev) or list of expressions (acc)
|
;; then: finished, return a number (prev) or list of expressions (acc)
|
||||||
(if (null? acc)
|
(if (null? acc)
|
||||||
|
|
|
@ -2,4 +2,5 @@ private
|
||||||
===
|
===
|
||||||
|
|
||||||
Files that no law-abiding library user should `require`.
|
Files that no law-abiding library user should `require`.
|
||||||
|
|
||||||
- `common.rkt` Helper functions common to a few macros.
|
- `common.rkt` Helper functions common to a few macros.
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
|
|
||||||
;; Common helper functions
|
;; Common helper functions
|
||||||
;; TODO actually detect the `quote` identifier, don't use eq?
|
|
||||||
|
|
||||||
(provide
|
(provide
|
||||||
expand-expr
|
expand-expr
|
||||||
|
@ -10,13 +9,13 @@
|
||||||
|
|
||||||
quoted-stx-value?
|
quoted-stx-value?
|
||||||
;; (-> Any (U #f Syntax))
|
;; (-> Any (U #f Syntax))
|
||||||
;; If the argument is a syntax object representing a quoted #%datum `v`,
|
;; If the argument is a syntax object representing a quoted datum `v`,
|
||||||
;; return `v`.
|
;; return `v`.
|
||||||
;; Otherwise, return #f.
|
;; Otherwise, return #f.
|
||||||
)
|
)
|
||||||
|
|
||||||
(require
|
(require
|
||||||
(for-template (only-in racket/base quote)))
|
(for-template (only-in typed/racket/base quote)))
|
||||||
|
|
||||||
;; =============================================================================
|
;; =============================================================================
|
||||||
|
|
||||||
|
|
13
regexp.rkt
13
regexp.rkt
|
@ -1,7 +1,9 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Stronger types for regular expression matching.
|
||||||
|
;;
|
||||||
;; TODO use syntax-class to abstract over local-expands / check num-groups
|
;; TODO use syntax-class to abstract over local-expands / check num-groups
|
||||||
;; TODO groups can be #f. Don't just 'error'
|
;; TODO groups can be #f when using | ... any other way?
|
||||||
|
|
||||||
(provide
|
(provide
|
||||||
regexp: define-regexp:
|
regexp: define-regexp:
|
||||||
|
@ -9,7 +11,8 @@
|
||||||
byte-regexp: define-byte-regexp:
|
byte-regexp: define-byte-regexp:
|
||||||
byte-pregexp: define-byte-pregexp:
|
byte-pregexp: define-byte-pregexp:
|
||||||
;; Expression and definition forms that try checking their argument patterns.
|
;; Expression and definition forms that try checking their argument patterns.
|
||||||
;; If check succeeds, will remember #groups for calls to `regexp-match:`.
|
;; If check succeeds, will remember the number of pattern groups
|
||||||
|
;; for calls to `regexp-match:`.
|
||||||
|
|
||||||
regexp-match:
|
regexp-match:
|
||||||
;; (-> Pattern String Any * (U #f (List String *N+1)))
|
;; (-> Pattern String Any * (U #f (List String *N+1)))
|
||||||
|
@ -17,14 +20,13 @@
|
||||||
;; If the pattern is determined statically, result will be either #f
|
;; If the pattern is determined statically, result will be either #f
|
||||||
;; or a list of N+1 strings, where N is the number of groups specified
|
;; or a list of N+1 strings, where N is the number of groups specified
|
||||||
;; the pattern.
|
;; the pattern.
|
||||||
;;
|
|
||||||
;; Will raise a compile-time exception if the pattern contains unmatched groups.
|
;; Will raise a compile-time exception if the pattern contains unmatched groups.
|
||||||
)
|
)
|
||||||
|
|
||||||
(require (for-syntax
|
(require (for-syntax
|
||||||
racket/base
|
typed/racket/base
|
||||||
(only-in racket/format ~a)
|
(only-in racket/format ~a)
|
||||||
racket/syntax
|
(only-in racket/syntax format-id)
|
||||||
syntax/id-table
|
syntax/id-table
|
||||||
syntax/parse
|
syntax/parse
|
||||||
trivial/private/common
|
trivial/private/common
|
||||||
|
@ -107,6 +109,7 @@
|
||||||
(format "Valid regexp pattern (contains unmatched ~a)" reason)
|
(format "Valid regexp pattern (contains unmatched ~a)" reason)
|
||||||
str))
|
str))
|
||||||
|
|
||||||
|
;; Dispatch for counting groups
|
||||||
(define-for-syntax (count-groups v-stx)
|
(define-for-syntax (count-groups v-stx)
|
||||||
(cond
|
(cond
|
||||||
[(syntax-property v-stx num-groups-key)
|
[(syntax-property v-stx num-groups-key)
|
||||||
|
|
|
@ -16,6 +16,8 @@
|
||||||
(printf: "hex ~o\n" (exact->inexact 0))
|
(printf: "hex ~o\n" (exact->inexact 0))
|
||||||
)))
|
)))
|
||||||
|
|
||||||
|
;; -----------------------------------------------------------------------------
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
(require
|
(require
|
||||||
rackunit)
|
rackunit)
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Successful use of `format:`
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
|
|
||||||
(require
|
(require
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
|
|
||||||
|
;; Math expressions that fail to typecheck
|
||||||
|
|
||||||
(define (expr->typed-module expr)
|
(define (expr->typed-module expr)
|
||||||
#`(module t typed/racket/base
|
#`(module t typed/racket/base
|
||||||
(require trivial/math)
|
(require trivial/math)
|
||||||
|
@ -19,6 +21,8 @@
|
||||||
(ann (/: 1 1 0) One)
|
(ann (/: 1 1 0) One)
|
||||||
)))
|
)))
|
||||||
|
|
||||||
|
;; -----------------------------------------------------------------------------
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
(require
|
(require
|
||||||
rackunit)
|
rackunit)
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Well-typed math
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
(require
|
(require
|
||||||
trivial/math
|
trivial/math
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
|
|
||||||
|
;; Ill-typed `regexp:` expressions
|
||||||
|
;;
|
||||||
;; TODO why can't I catch errors for (ann ... (List String))? WhydoI need #f?
|
;; TODO why can't I catch errors for (ann ... (List String))? WhydoI need #f?
|
||||||
|
|
||||||
(define (expr->typed-module expr)
|
(define (expr->typed-module expr)
|
||||||
|
@ -36,6 +38,8 @@
|
||||||
(U #f (List String String)))
|
(U #f (List String String)))
|
||||||
)))
|
)))
|
||||||
|
|
||||||
|
;; -----------------------------------------------------------------------------
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
(require
|
(require
|
||||||
rackunit)
|
rackunit)
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#lang typed/racket/base
|
#lang typed/racket/base
|
||||||
|
|
||||||
|
;; Well-typed use of regexp:
|
||||||
|
|
||||||
(module+ test
|
(module+ test
|
||||||
|
|
||||||
(require
|
(require
|
||||||
|
|
Loading…
Reference in New Issue
Block a user