mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 09:03:42 -05:00
parent
a628a499fa
commit
0c324a442e
6 changed files with 156 additions and 8 deletions
|
@ -13,6 +13,12 @@ import { assert, copyBytes } from "./util.ts";
|
||||||
const MIN_READ = 512;
|
const MIN_READ = 512;
|
||||||
const MAX_SIZE = 2 ** 32 - 2;
|
const MAX_SIZE = 2 ** 32 - 2;
|
||||||
|
|
||||||
|
const encoder = new TextEncoder();
|
||||||
|
export function stringsReader(s: string): Reader {
|
||||||
|
const ui8 = encoder.encode(s);
|
||||||
|
return new Buffer(ui8.buffer as ArrayBuffer);
|
||||||
|
}
|
||||||
|
|
||||||
/** A Buffer is a variable-sized buffer of bytes with read() and write()
|
/** A Buffer is a variable-sized buffer of bytes with read() and write()
|
||||||
* methods. Based on https://golang.org/pkg/bytes/#Buffer
|
* methods. Based on https://golang.org/pkg/bytes/#Buffer
|
||||||
*/
|
*/
|
||||||
|
|
2
bufio.ts
2
bufio.ts
|
@ -1,4 +1,4 @@
|
||||||
// Ported to Deno from:
|
// Based on https://github.com/golang/go/blob/891682/src/bufio/bufio.go
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
// Copyright 2009 The Go Authors. All rights reserved.
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
// Ported to Deno from:
|
// Based on https://github.com/golang/go/blob/891682/src/bufio/bufio_test.go
|
||||||
// Copyright 2009 The Go Authors. All rights reserved.
|
// Copyright 2009 The Go Authors. All rights reserved.
|
||||||
// Use of this source code is governed by a BSD-style
|
// Use of this source code is governed by a BSD-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
@ -10,7 +10,7 @@ import {
|
||||||
assertEqual
|
assertEqual
|
||||||
} from "https://deno.land/x/testing/testing.ts";
|
} from "https://deno.land/x/testing/testing.ts";
|
||||||
import { BufReader, BufState } from "./bufio.ts";
|
import { BufReader, BufState } from "./bufio.ts";
|
||||||
import { Buffer } from "./buffer.ts";
|
import { Buffer, stringsReader } from "./buffer.ts";
|
||||||
import * as iotest from "./iotest.ts";
|
import * as iotest from "./iotest.ts";
|
||||||
import { charCode, copyBytes } from "./util.ts";
|
import { charCode, copyBytes } from "./util.ts";
|
||||||
|
|
||||||
|
@ -31,11 +31,6 @@ async function readBytes(buf: BufReader): Promise<string> {
|
||||||
return decoder.decode(b.subarray(0, nb));
|
return decoder.decode(b.subarray(0, nb));
|
||||||
}
|
}
|
||||||
|
|
||||||
function stringsReader(s: string): Reader {
|
|
||||||
const ui8 = encoder.encode(s);
|
|
||||||
return new Buffer(ui8.buffer as ArrayBuffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
test(async function bufioReaderSimple() {
|
test(async function bufioReaderSimple() {
|
||||||
const data = "hello world";
|
const data = "hello world";
|
||||||
const b = new BufReader(stringsReader(data));
|
const b = new BufReader(stringsReader(data));
|
||||||
|
|
1
test.ts
1
test.ts
|
@ -1,3 +1,4 @@
|
||||||
import "./buffer_test.ts";
|
import "./buffer_test.ts";
|
||||||
import "./bufio_test.ts";
|
import "./bufio_test.ts";
|
||||||
|
import "./textproto_test.ts";
|
||||||
// TODO import "./http_test.ts";
|
// TODO import "./http_test.ts";
|
||||||
|
|
144
textproto.ts
Normal file
144
textproto.ts
Normal file
|
@ -0,0 +1,144 @@
|
||||||
|
// Based on https://github.com/golang/go/blob/891682/src/net/textproto/
|
||||||
|
// Copyright 2009 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
import { BufReader, BufState } from "./bufio.ts";
|
||||||
|
import { charCode } from "./util.ts";
|
||||||
|
|
||||||
|
const asciiDecoder = new TextDecoder("ascii");
|
||||||
|
function str(buf: Uint8Array): string {
|
||||||
|
if (buf == null) {
|
||||||
|
return "";
|
||||||
|
} else {
|
||||||
|
return asciiDecoder.decode(buf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ProtocolError extends Error {
|
||||||
|
constructor(msg: string) {
|
||||||
|
super(msg);
|
||||||
|
this.name = "ProtocolError";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TextProtoReader {
|
||||||
|
constructor(readonly r: BufReader) {}
|
||||||
|
|
||||||
|
/** readLine() reads a single line from the TextProtoReader,
|
||||||
|
* eliding the final \n or \r\n from the returned string.
|
||||||
|
*/
|
||||||
|
async readLine(): Promise<[string, BufState]> {
|
||||||
|
let [line, err] = await this.readLineSlice();
|
||||||
|
return [str(line), err];
|
||||||
|
}
|
||||||
|
|
||||||
|
/** ReadMIMEHeader reads a MIME-style header from r.
|
||||||
|
* The header is a sequence of possibly continued Key: Value lines
|
||||||
|
* ending in a blank line.
|
||||||
|
* The returned map m maps CanonicalMIMEHeaderKey(key) to a
|
||||||
|
* sequence of values in the same order encountered in the input.
|
||||||
|
*
|
||||||
|
* For example, consider this input:
|
||||||
|
*
|
||||||
|
* My-Key: Value 1
|
||||||
|
* Long-Key: Even
|
||||||
|
* Longer Value
|
||||||
|
* My-Key: Value 2
|
||||||
|
*
|
||||||
|
* Given that input, ReadMIMEHeader returns the map:
|
||||||
|
*
|
||||||
|
* map[string][]string{
|
||||||
|
* "My-Key": {"Value 1", "Value 2"},
|
||||||
|
* "Long-Key": {"Even Longer Value"},
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
async readMIMEHeader(): Promise<Headers> {
|
||||||
|
let m = new Headers();
|
||||||
|
let line: Uint8Array;
|
||||||
|
|
||||||
|
// The first line cannot start with a leading space.
|
||||||
|
let [buf, err] = await this.r.peek(1);
|
||||||
|
if (buf[0] == charCode(' ') || buf[0] == charCode('\t')) {
|
||||||
|
[line, err] = await this.readLineSlice();
|
||||||
|
}
|
||||||
|
|
||||||
|
[buf, err] = await this.r.peek(1);
|
||||||
|
if (err == null && (buf[0] == charCode(' ') || buf[0] == charCode('\t'))) {
|
||||||
|
throw new ProtocolError(`malformed MIME header initial line: ${str(line)}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
let [kv, err] = await this.readLineSlice(); // readContinuedLineSlice
|
||||||
|
if (kv.byteLength == 0) {
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Key ends at first colon; should not have trailing spaces
|
||||||
|
// but they appear in the wild, violating specs, so we remove
|
||||||
|
// them if present.
|
||||||
|
let i = kv.indexOf(charCode(':'));
|
||||||
|
if (i < 0) {
|
||||||
|
throw new ProtocolError(`malformed MIME header line: ${str(kv)}`);
|
||||||
|
}
|
||||||
|
let endKey = i;
|
||||||
|
while (endKey > 0 && kv[endKey - 1] == charCode(' ')) {
|
||||||
|
endKey--;
|
||||||
|
}
|
||||||
|
|
||||||
|
//let key = canonicalMIMEHeaderKey(kv.subarray(0, endKey));
|
||||||
|
let key = str(kv.subarray(0, endKey));
|
||||||
|
|
||||||
|
// As per RFC 7230 field-name is a token, tokens consist of one or more chars.
|
||||||
|
// We could return a ProtocolError here, but better to be liberal in what we
|
||||||
|
// accept, so if we get an empty key, skip it.
|
||||||
|
if (key == "") {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip initial spaces in value.
|
||||||
|
i++; // skip colon
|
||||||
|
while (i < kv.byteLength &&
|
||||||
|
(kv[i] == charCode(' ') || kv[i] == charCode('\t'))) {
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
let value = str(kv.subarray(i));
|
||||||
|
|
||||||
|
m.append(key, value);
|
||||||
|
|
||||||
|
if (err != null) {
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
async readLineSlice(): Promise<[Uint8Array, BufState]> {
|
||||||
|
// this.closeDot();
|
||||||
|
let line: null | Uint8Array;
|
||||||
|
while (true) {
|
||||||
|
let [l, more, err] = await this.r.readLine();
|
||||||
|
if (err != null) {
|
||||||
|
return [null, err];
|
||||||
|
}
|
||||||
|
// Avoid the copy if the first call produced a full line.
|
||||||
|
if (line == null && !more) {
|
||||||
|
return [l, null];
|
||||||
|
}
|
||||||
|
line = append(line, l);
|
||||||
|
if (!more) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return [line, null];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function append(a: Uint8Array, b: Uint8Array): Uint8Array {
|
||||||
|
if (a == null) {
|
||||||
|
return b;
|
||||||
|
} else {
|
||||||
|
throw Error("Not implemented");
|
||||||
|
}
|
||||||
|
}
|
2
util.ts
2
util.ts
|
@ -1,3 +1,5 @@
|
||||||
|
import { Reader } from "deno";
|
||||||
|
|
||||||
export function assert(cond: boolean, msg = "assert") {
|
export function assert(cond: boolean, msg = "assert") {
|
||||||
if (!cond) {
|
if (!cond) {
|
||||||
throw Error(msg);
|
throw Error(msg);
|
||||||
|
|
Loading…
Reference in a new issue