mirror of
https://github.com/denoland/deno.git
synced 2025-10-02 07:01:16 +00:00
Reorgnanize repos, examples and tests (denoland/deno_std#105)
Original: c5e6e015b5
This commit is contained in:
parent
7d6a0f64f2
commit
f626b04ebe
44 changed files with 54 additions and 45 deletions
150
textproto/mod.ts
Normal file
150
textproto/mod.ts
Normal file
|
@ -0,0 +1,150 @@
|
|||
// 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 "../io/bufio.ts";
|
||||
import { charCode } from "../io/util.ts";
|
||||
|
||||
const asciiDecoder = new TextDecoder();
|
||||
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, BufState]> {
|
||||
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, err];
|
||||
}
|
||||
|
||||
// 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: Uint8Array;
|
||||
while (true) {
|
||||
let [l, more, err] = await this.r.readLine();
|
||||
if (err != null) {
|
||||
// Go's len(typed nil) works fine, but not in JS
|
||||
return [new Uint8Array(0), 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];
|
||||
}
|
||||
}
|
||||
|
||||
export function append(a: Uint8Array, b: Uint8Array): Uint8Array {
|
||||
if (a == null) {
|
||||
return b;
|
||||
} else {
|
||||
const output = new Uint8Array(a.length + b.length);
|
||||
output.set(a, 0);
|
||||
output.set(b, a.length);
|
||||
return output;
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue