2019-08-15 10:43:13 +00:00
|
|
|
import { Stream } from 'stream';
|
2021-11-27 08:40:44 +00:00
|
|
|
import is from '@sindresorhus/is';
|
2020-05-01 16:03:48 +00:00
|
|
|
import bunyan from 'bunyan';
|
|
|
|
import fs from 'fs-extra';
|
2020-10-26 12:58:28 +00:00
|
|
|
import { clone } from '../util/clone';
|
2021-05-11 10:51:21 +00:00
|
|
|
import { HttpError } from '../util/http/types';
|
2020-05-16 10:35:41 +00:00
|
|
|
import { redactedFields, sanitize } from '../util/sanitize';
|
2021-05-11 10:51:21 +00:00
|
|
|
import type { BunyanRecord, BunyanStream } from './types';
|
2019-08-15 10:43:13 +00:00
|
|
|
|
|
|
|
const excludeProps = ['pid', 'time', 'v', 'hostname'];
|
|
|
|
|
2020-09-21 20:04:11 +00:00
|
|
|
export class ProblemStream extends Stream {
|
|
|
|
private _problems: BunyanRecord[] = [];
|
2019-08-15 10:43:13 +00:00
|
|
|
|
|
|
|
readable: boolean;
|
|
|
|
|
|
|
|
writable: boolean;
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.readable = false;
|
|
|
|
this.writable = true;
|
|
|
|
}
|
|
|
|
|
2019-11-26 15:13:07 +00:00
|
|
|
write(data: BunyanRecord): boolean {
|
2020-09-21 20:04:11 +00:00
|
|
|
const problem = { ...data };
|
2020-03-17 11:15:22 +00:00
|
|
|
for (const prop of excludeProps) {
|
2020-09-21 20:04:11 +00:00
|
|
|
delete problem[prop];
|
2020-03-17 11:15:22 +00:00
|
|
|
}
|
2020-09-21 20:04:11 +00:00
|
|
|
this._problems.push(problem);
|
2019-08-15 10:43:13 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-09-21 20:04:11 +00:00
|
|
|
getProblems(): BunyanRecord[] {
|
|
|
|
return this._problems;
|
2019-08-15 10:43:13 +00:00
|
|
|
}
|
2020-09-18 07:26:38 +00:00
|
|
|
|
2020-09-21 20:04:11 +00:00
|
|
|
clearProblems(): void {
|
|
|
|
this._problems = [];
|
2020-09-18 07:26:38 +00:00
|
|
|
}
|
2019-08-15 10:43:13 +00:00
|
|
|
}
|
2020-05-16 10:35:41 +00:00
|
|
|
const templateFields = ['prBody'];
|
|
|
|
const contentFields = [
|
|
|
|
'content',
|
|
|
|
'contents',
|
|
|
|
'packageLockParsed',
|
|
|
|
'yarnLockParsed',
|
|
|
|
];
|
2019-09-27 09:28:09 +00:00
|
|
|
|
2020-10-26 12:58:28 +00:00
|
|
|
export default function prepareError(err: Error): Record<string, unknown> {
|
|
|
|
const response: Record<string, unknown> = {
|
|
|
|
...err,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Required as message is non-enumerable
|
|
|
|
if (!response.message && err.message) {
|
|
|
|
response.message = err.message;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Required as stack is non-enumerable
|
|
|
|
if (!response.stack && err.stack) {
|
|
|
|
response.stack = err.stack;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle got error
|
2021-05-11 10:51:21 +00:00
|
|
|
if (err instanceof HttpError) {
|
2020-10-26 12:58:28 +00:00
|
|
|
const options: Record<string, unknown> = {
|
|
|
|
headers: clone(err.options.headers),
|
|
|
|
url: err.options.url?.toString(),
|
2021-09-01 11:07:55 +00:00
|
|
|
hostType: err.options.context.hostType,
|
2020-10-26 12:58:28 +00:00
|
|
|
};
|
|
|
|
response.options = options;
|
|
|
|
|
2021-11-27 09:22:58 +00:00
|
|
|
options.username = err.options.username;
|
|
|
|
options.password = err.options.password;
|
|
|
|
options.method = err.options.method;
|
|
|
|
options.http2 = err.options.http2;
|
2020-10-26 12:58:28 +00:00
|
|
|
|
|
|
|
// istanbul ignore else
|
|
|
|
if (err.response) {
|
|
|
|
response.response = {
|
|
|
|
statusCode: err.response?.statusCode,
|
|
|
|
statusMessage: err.response?.statusMessage,
|
2021-11-17 08:55:57 +00:00
|
|
|
body:
|
|
|
|
err.name === 'TimeoutError' ? undefined : clone(err.response.body),
|
2020-10-26 12:58:28 +00:00
|
|
|
headers: clone(err.response.headers),
|
|
|
|
httpVersion: err.response.httpVersion,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
type NestedValue = unknown[] | object;
|
|
|
|
|
|
|
|
function isNested(value: unknown): value is NestedValue {
|
|
|
|
return is.array(value) || is.object(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function sanitizeValue(
|
|
|
|
value: unknown,
|
|
|
|
seen = new WeakMap<NestedValue, unknown>()
|
|
|
|
): any {
|
|
|
|
if (is.string(value)) {
|
|
|
|
return sanitize(value);
|
2019-09-27 09:28:09 +00:00
|
|
|
}
|
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
if (is.date(value)) {
|
|
|
|
return value;
|
2020-05-16 10:35:41 +00:00
|
|
|
}
|
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
if (is.function_(value)) {
|
|
|
|
return '[function]';
|
2020-10-26 12:58:28 +00:00
|
|
|
}
|
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
if (is.buffer(value)) {
|
|
|
|
return '[content]';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is.error(value)) {
|
|
|
|
const err = prepareError(value);
|
|
|
|
return sanitizeValue(err, seen);
|
|
|
|
}
|
2019-09-27 09:28:09 +00:00
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
if (is.array(value)) {
|
|
|
|
const length = value.length;
|
|
|
|
const arrayResult = Array(length);
|
|
|
|
seen.set(value, arrayResult);
|
|
|
|
for (let idx = 0; idx < length; idx += 1) {
|
|
|
|
const val = value[idx];
|
|
|
|
arrayResult[idx] =
|
|
|
|
isNested(val) && seen.has(val)
|
|
|
|
? seen.get(val)
|
|
|
|
: sanitizeValue(val, seen);
|
2019-11-15 15:04:58 +00:00
|
|
|
}
|
2021-11-27 08:40:44 +00:00
|
|
|
return arrayResult;
|
|
|
|
}
|
2019-11-15 15:04:58 +00:00
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
if (is.object(value)) {
|
2019-09-27 09:28:09 +00:00
|
|
|
const objectResult: Record<string, any> = {};
|
2021-11-27 08:40:44 +00:00
|
|
|
seen.set(value, objectResult);
|
2019-09-27 09:28:09 +00:00
|
|
|
for (const [key, val] of Object.entries<any>(value)) {
|
2020-05-16 10:35:41 +00:00
|
|
|
let curValue: any;
|
2020-10-28 10:38:28 +00:00
|
|
|
if (!val) {
|
|
|
|
curValue = val;
|
|
|
|
} else if (redactedFields.includes(key)) {
|
2020-05-16 10:35:41 +00:00
|
|
|
curValue = '***********';
|
|
|
|
} else if (contentFields.includes(key)) {
|
|
|
|
curValue = '[content]';
|
|
|
|
} else if (templateFields.includes(key)) {
|
|
|
|
curValue = '[Template]';
|
2021-03-22 14:51:38 +00:00
|
|
|
} else if (key === 'secrets') {
|
|
|
|
curValue = {};
|
|
|
|
Object.keys(val).forEach((secretKey) => {
|
|
|
|
curValue[secretKey] = '***********';
|
|
|
|
});
|
2020-05-16 10:35:41 +00:00
|
|
|
} else {
|
|
|
|
curValue = seen.has(val) ? seen.get(val) : sanitizeValue(val, seen);
|
|
|
|
}
|
|
|
|
|
|
|
|
objectResult[key] = curValue;
|
2019-09-27 09:28:09 +00:00
|
|
|
}
|
2021-11-27 08:40:44 +00:00
|
|
|
|
2019-09-27 09:28:09 +00:00
|
|
|
return objectResult;
|
|
|
|
}
|
|
|
|
|
2021-11-27 08:40:44 +00:00
|
|
|
return value;
|
2019-09-27 09:28:09 +00:00
|
|
|
}
|
|
|
|
|
2020-08-27 07:12:37 +00:00
|
|
|
export function withSanitizer(streamConfig: bunyan.Stream): bunyan.Stream {
|
2020-03-17 11:15:22 +00:00
|
|
|
if (streamConfig.type === 'rotating-file') {
|
2019-09-27 09:28:09 +00:00
|
|
|
throw new Error("Rotating files aren't supported");
|
2020-03-17 11:15:22 +00:00
|
|
|
}
|
2019-09-27 09:28:09 +00:00
|
|
|
|
2020-08-27 07:12:37 +00:00
|
|
|
const stream = streamConfig.stream as BunyanStream;
|
2020-07-18 06:42:32 +00:00
|
|
|
if (stream?.writable) {
|
2021-11-27 09:22:58 +00:00
|
|
|
const write = (
|
|
|
|
chunk: BunyanRecord,
|
|
|
|
enc: BufferEncoding,
|
|
|
|
cb: (err?: Error | null) => void
|
|
|
|
): void => {
|
2019-09-27 09:28:09 +00:00
|
|
|
const raw = sanitizeValue(chunk);
|
|
|
|
const result =
|
|
|
|
streamConfig.type === 'raw'
|
|
|
|
? raw
|
2021-11-29 19:16:05 +00:00
|
|
|
: JSON.stringify(raw, bunyan.safeCycles()).replace(/\n?$/, '\n'); // TODO #12874
|
2019-09-27 09:28:09 +00:00
|
|
|
stream.write(result, enc, cb);
|
|
|
|
};
|
|
|
|
|
|
|
|
return {
|
|
|
|
...streamConfig,
|
|
|
|
type: 'raw',
|
|
|
|
stream: { write },
|
2020-08-27 07:12:37 +00:00
|
|
|
} as bunyan.Stream;
|
2019-09-27 09:28:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (streamConfig.path) {
|
|
|
|
const fileStream = fs.createWriteStream(streamConfig.path, {
|
|
|
|
flags: 'a',
|
|
|
|
encoding: 'utf8',
|
|
|
|
});
|
|
|
|
|
|
|
|
return withSanitizer({ ...streamConfig, stream: fileStream });
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error("Missing 'stream' or 'path' for bunyan stream");
|
|
|
|
}
|