** update inline comments

This commit is contained in:
ben 2015-12-14 02:32:47 -05:00
parent 7e30256efc
commit 03899c77f1
13 changed files with 56 additions and 24 deletions

View File

@ -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

View File

@ -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)])))

View File

@ -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")))

View File

@ -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)

View File

@ -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.

View File

@ -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)))
;; ============================================================================= ;; =============================================================================

View File

@ -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)

View File

@ -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)

View File

@ -1,5 +1,7 @@
#lang typed/racket/base #lang typed/racket/base
;; Successful use of `format:`
(module+ test (module+ test
(require (require

View File

@ -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)

View File

@ -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

View File

@ -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)

View File

@ -1,5 +1,7 @@
#lang typed/racket/base #lang typed/racket/base
;; Well-typed use of regexp:
(module+ test (module+ test
(require (require