|
5 | 5 | [clojure.tools.reader.reader-types :as r]
|
6 | 6 | [rewrite-clj.interop :as interop])
|
7 | 7 | #?(:cljs (:import [goog.string StringBuffer])
|
8 |
| - :clj (:import [java.io PushbackReader] |
| 8 | + :clj (:import [java.io PushbackReader Closeable] |
9 | 9 | [clojure.tools.reader.reader_types IndexingPushbackReader])))
|
10 | 10 |
|
11 | 11 | #?(:clj (set! *warn-on-reflection* true))
|
|
171 | 171 |
|
172 | 172 | ;; ## Reader Types
|
173 | 173 |
|
174 |
| -(defn string-reader |
175 |
| - "Create reader for strings." |
176 |
| - [s] |
177 |
| - (r/indexing-push-back-reader |
178 |
| - (r/string-push-back-reader s))) |
| 174 | +;; |
| 175 | +;; clojure.tools.reader (at the time of this writing v1.3.5) does not seem to normalize Windows \r\n newlines |
| 176 | +;; properly to \n for Clojure |
| 177 | +;; |
| 178 | +;; ClojureScript seems to work fine - but note that for peek it can return \r instead of \n. |
| 179 | +;; |
| 180 | +;; see https://clojure.atlassian.net/browse/TRDR-65 |
| 181 | +;; |
| 182 | +;; For now, we introduce a normalizing reader for Clojure. |
| 183 | +;; Once/if this isssue is fixed in in tools reader we can turf our work-around. |
| 184 | + |
| 185 | +#?(:clj |
| 186 | + (deftype NewlineNormalizingReader |
| 187 | + [rdr |
| 188 | + ^:unsynchronized-mutable next-char |
| 189 | + ^:unsynchronized-mutable peeked-char] |
| 190 | + r/Reader |
| 191 | + (read-char [_reader] |
| 192 | + (if peeked-char |
| 193 | + (let [ch peeked-char] |
| 194 | + (set! peeked-char nil) |
| 195 | + ch) |
| 196 | + (let [ch (or next-char (r/read-char rdr))] |
| 197 | + (when next-char (set! next-char nil)) |
| 198 | + (cond (identical? \return ch) |
| 199 | + (let [next-ch (r/read-char rdr)] |
| 200 | + (when (not (identical? \newline next-ch)) |
| 201 | + (set! next-char next-ch)) |
| 202 | + \newline) |
| 203 | + |
| 204 | + (identical? \formfeed ch) |
| 205 | + \newline |
| 206 | + |
| 207 | + :else |
| 208 | + ch)))) |
| 209 | + |
| 210 | + (peek-char [reader] |
| 211 | + (let [ch (or peeked-char (.read-char reader))] |
| 212 | + (when-not peeked-char (set! peeked-char ch)) |
| 213 | + ch)))) |
| 214 | + |
| 215 | +#?(:clj |
| 216 | + (defn ^Closeable newline-normalizing-reader |
| 217 | + "Normalizes the following line endings to LF (line feed - 0x0A): |
| 218 | + - LF (remains LF) |
| 219 | + - CRLF (carriage return 0x0D line feed 0x0A) |
| 220 | + - FF (form feed 0x0C)" |
| 221 | + [rdr] |
| 222 | + (NewlineNormalizingReader. (r/to-rdr rdr) nil nil))) |
179 | 223 |
|
180 | 224 | #?(:clj
|
181 | 225 | (defn file-reader
|
|
185 | 229 | (-> (io/file f)
|
186 | 230 | (io/reader)
|
187 | 231 | (PushbackReader. 2)
|
| 232 | + newline-normalizing-reader |
188 | 233 | (r/indexing-push-back-reader 2))))
|
| 234 | + |
| 235 | +(defn string-reader |
| 236 | + "Create reader for strings." |
| 237 | + [s] |
| 238 | + (-> s |
| 239 | + r/string-push-back-reader |
| 240 | + #?@(:clj [newline-normalizing-reader]) |
| 241 | + r/indexing-push-back-reader)) |
0 commit comments