;;; CHICKEN Transducers - Transducers for working with foldable data types. ;;; ;;; Copyright (c) 2023 Jeremy Steward ;;; ;;; Permission is hereby granted, free of charge, to any person obtaining a copy ;;; of this software and associated documentation files (the "Software"), to deal ;;; in the Software without restriction, including without limitation the rights ;;; to use, copy, modify, merge, publish, distribute, sublicense, and/or sell ;;; copies of the Software, and to permit persons to whom the Software is ;;; furnished to do so, subject to the following conditions: ;;; ;;; The above copyright notice and this permission notice shall be included in all ;;; copies or substantial portions of the Software. ;;; ;;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR ;;; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, ;;; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE ;;; AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER ;;; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, ;;; OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE ;;; SOFTWARE. (define-library (transducers base) (import (except (scheme base) map for-each) (scheme case-lambda) (srfi 143)) (cond-expand (chicken-5 (import (only (chicken base) void compose)) (import-syntax (srfi 253)) (import-syntax (chicken type))) (else ;; Unlike in CHICKEN, there is no import-syntax in r7rs generally (import (srfi 253)) (define compose values) (define unspecified-value (list '*UNSPECIFIED*)) (define (void) unspecified-value))) (include-library-declarations "src/transducers.base.exports.scm") (begin ;; Predicate to check for a non-negative (zero or greater) fixnum (define (non-negative-fixnum? x) (and (fixnum? x) (fx>=? x 0))) ;; Predicate to check for a positive (one or greater) fixnum (define (positive-fixnum? x) (and (fixnum? x) (positive? x))) ;; A record type expressing the concept of a reduced value - i.e. a value ;; that tells the transduction to stop due to a completed step in the ;; transducer. ;; ;; This is needed to be separate from "exhausted," which represents a ;; collection that has no more remaining values. (define-record-type ;; Constructor for creating a `reduced` value. (make-reduced value) ;; Predicate to check if something is a reduced value. reduced? ;; Unwraps a reduced value and returns the value. (value unwrap)) ;; A reducer that aggregates all items as if applying `(or (pred? <>) ...)` across ;; all items transduced over. (define collect-any (case-lambda ((pred?) (collect-any pred? #f)) ((pred? sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (let ((test (pred? item))) (and (or result test) (make-reduced test)))))))) ;; A reducer that aggregates all items as if applying `(and (pred? <>) ...)` ;; across all items transduced over. (define collect-all (case-lambda ((pred?) (collect-all pred? #t)) ((pred? sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (let ((test (pred? item))) (if (and result test) test (make-reduced #f)))))))) ;; A reducer that aggregates the total number of items transduced over. (define collect-count (case-lambda (() (collect-count 0)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (fx+ result 1)))))) ;; A reducer that aggregates the maximum of the items transduced over. (define collect-max (case-lambda (() (collect-max -inf.0)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (max result item)))))) ;; A reducer that aggregates the minimum of the items transduced over. (define collect-min (case-lambda (() (collect-min +inf.0)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (min result item)))))) ;; A reducer that returns just the first item, or #f if there are no items. (define collect-first (case-lambda (() (collect-first #f)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (make-reduced item)))))) ;; A reducer that returns just the last item, or #f if there are no items. (define collect-last (case-lambda (() (collect-last #f)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) item))))) ;; A reducer that returns the sum of the items using `+` (define collect-sum (case-lambda (() (collect-sum 0)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (+ result item)))))) ;; A reducer that returns the product of the items using `*` (define collect-product (case-lambda (() (collect-product 1)) ((sentinel) (case-lambda (() sentinel) ((result) result) ((result item) (* result item)))))) ;; A reducer that collects all items but always returns an undefined result ;; ;; Primarily useful for implementing for-each, which is just a transducer ;; that operates over all items in an iterable but then returns an undefined ;; result at the end. (define collect-void (case-lambda (() (collect-void (void))) ((_sentinel) (case-lambda (() (void)) ((result) result) ((result item) result))))) ;; A procedure that transduces over an iterable with the provided xform, ;; collector, and iterable. (define (transduce folder transducer collector data) (let* ((xf (transducer collector)) (result (folder xf (collector) data))) (xf result))) ;; A folding procedure that folds over exactly one item — This is useful ;; for testing a set of transducers / collectors as well as just using ;; transducers to operate over a single item without wrapping it in a list ;; or something silly like that. (define (once-folder xs sentinel item) (let ((result (xs sentinel item))) (if (reduced? result) (unwrap result) result))) ;; A transduce operation that operates exactly on a single item. (define (once transducer collector item) (transduce once-folder transducer collector item)) ;; Transducer that performs a mapping operation over each item in the iterable. ;; ;; For every item that passes through this transducer, (f item) is passed on ;; to the reducing function. (define (map f) (lambda (reducer) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (reducer result (f item)))))) ;; Transducer that performs a filtering operation over each item in the iterable. ;; ;; For every item that passes through this transducer, the item is passed ;; forward iff it satisfies `pred` (define (filter pred) (lambda (reducer) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (if (pred item) (reducer result item) result))))) ;; Transducer that operates like filter, but will only forward items iff ;; the item satisfies `(compose not pred)`. (define (remove pred) (filter (compose not pred))) ;; A transducer that drops `n` number of items before forwarding all ;; remaining items. (define-checked (drop (n non-negative-fixnum?)) (lambda (reducer) (let ((n n)) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (set! n (fx- n 1)) (if (fx>=? n 0) result (reducer result item))))))) ;; A transducer that drops items while pred is true. After the first instance ;; of pred returning false, continues to forward items along to the reducer. (define (drop-while pred) (lambda (reducer) (let ((continue-to-drop #t)) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (if continue-to-drop (cond ((pred item) result) (else (set! continue-to-drop #f) (reducer result item))) (reducer result item))))))) ;; A transducer that takes `n` number of items before dropping all remaining ;; items. (define-checked (take (n positive-fixnum?)) (lambda (reducer) (let ((n n)) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (set! n (fx- n 1)) (if (fx>? n 0) (reducer result item) (make-reduced (reducer result item)))))))) ;; A transducer that takes items as long as `pred` is true, and then drops ;; all remaining items after `pred` first turns false. (define (take-while pred) (lambda (reducer) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (if (pred item) (reducer result item) (make-reduced result)))))) ;; Helper function that wraps a reduced value twice as traversing full ;; iterables will unwrap them. `flatten` is a good example: it re-uses its ;; reducer on the item using `traverse-iterable.` If that reduction finishes ;; early and returns a reduced value, then it will be immediately unwrapped ;; and we will want to keep moving forward with the transduction. (define (preserving-reduced reducer) (lambda (acc x) (let ((ret (reducer acc x))) (if (reduced? ret) (make-reduced ret) ret)))) ;; Defines a transducer that flattens every item that satisfies `type?` by ;; folding over that type. (define-syntax define-flatten-transducer (syntax-rules () ((_ name type? type-fold) (define name (lambda (reducer) (case-lambda (() (reducer)) ((result) (reducer result)) ((result item) (if (type? item) (type-fold (preserving-reduced (name reducer)) result item) (reducer result item))))))))) ;; Chains the iterable into the transduction, as if appending one iterable to ;; another. (define-syntax define-chain-transducer (syntax-rules () ((_ name type-fold) (define (name iterable) (lambda (reducer) (case-lambda (() (reducer)) ((result) (reducer (type-fold reducer result iterable))) ((result item) (reducer result item)))))))) ;; Collects items into a collection of `n` items defined by `collector`. ;; ;; Collector can be any valid collector, e.g. collect-list, every, any, +, *, ;; etc. ;; ;; NOTE: chunks will collect up-to `n` items in each collection. If the end ;; of a collection is reached before `n` items are collected, then the last ;; chunk will only have as many items (less than `n`) that were collected. (define-checked (chunks (n positive-fixnum?) collector) (lambda (reducer) (let ((i 0) (collection (collector))) (case-lambda (() (reducer)) ((result) ;; If there is anything remaining in our collector when finalizing we ;; collect here. This last collection will have `i` number of items. (if (fx>? i 0) (let ((new-result (reducer result (collector collection)))) (if (reduced? new-result) (reducer (unwrap new-result)) (reducer new-result))) (reducer result))) ((result item) (let ((new-i (fx+ i 1)) (new-collection (collector collection item))) (cond ((fx ;; Constructs a transducible type-class. (make-transducible folder collector flattener chainer interleaver zipper) ;; Predicate for testing if an object is a transducible type-class transducible? ;; A procedure of the form (fold f sentinel xs) that defines the folding ;; operation over a transducible sequence. (folder transducible-folder) ;; A procedure of the form (collect #!optional sentinel) that defines the ;; operation that collects a transduction into a transducible sequence. (collector transducible-collector) ;; A procedure of the form (flatten xs) that defines the transducer that ;; flattens transducible sequences from elements in the transduction. (flattener transducible-flattener) ;; A procedure of the form (chain xs) that appends the elements of a ;; transducible sequence into the transduction. (chainer transducible-chainer) ;; A procedure of the form (interleave xs) that interleaves elements of a ;; transducible sequence into the transduction. (interleaver transducible-interleaver) ;; A procedure of the form (zip xs) that zips elements of a transducible ;; sequence into the transduction. (zipper transducible-zipper))) ;; End-of-module )