mirror of
https://github.com/denoland/deno.git
synced 2025-01-21 13:00:36 -05:00
feat(unstable): Instrument fetch (#27057)
Add basic tracing to `fetch`. Also fix span kinds so that we can differentiate fetch and serve.
This commit is contained in:
parent
08a56763d4
commit
d59bd5e8c9
5 changed files with 223 additions and 90 deletions
|
@ -10,9 +10,10 @@
|
||||||
/// <reference path="./lib.deno_fetch.d.ts" />
|
/// <reference path="./lib.deno_fetch.d.ts" />
|
||||||
/// <reference lib="esnext" />
|
/// <reference lib="esnext" />
|
||||||
|
|
||||||
import { core, primordials } from "ext:core/mod.js";
|
import { core, internals, primordials } from "ext:core/mod.js";
|
||||||
import {
|
import {
|
||||||
op_fetch,
|
op_fetch,
|
||||||
|
op_fetch_promise_is_settled,
|
||||||
op_fetch_send,
|
op_fetch_send,
|
||||||
op_wasm_streaming_feed,
|
op_wasm_streaming_feed,
|
||||||
op_wasm_streaming_set_url,
|
op_wasm_streaming_set_url,
|
||||||
|
@ -28,7 +29,9 @@ const {
|
||||||
PromisePrototypeThen,
|
PromisePrototypeThen,
|
||||||
PromisePrototypeCatch,
|
PromisePrototypeCatch,
|
||||||
SafeArrayIterator,
|
SafeArrayIterator,
|
||||||
|
SafePromisePrototypeFinally,
|
||||||
String,
|
String,
|
||||||
|
StringPrototypeSlice,
|
||||||
StringPrototypeStartsWith,
|
StringPrototypeStartsWith,
|
||||||
StringPrototypeToLowerCase,
|
StringPrototypeToLowerCase,
|
||||||
TypeError,
|
TypeError,
|
||||||
|
@ -307,93 +310,150 @@ function httpRedirectFetch(request, response, terminator) {
|
||||||
* @param {RequestInit} init
|
* @param {RequestInit} init
|
||||||
*/
|
*/
|
||||||
function fetch(input, init = { __proto__: null }) {
|
function fetch(input, init = { __proto__: null }) {
|
||||||
// There is an async dispatch later that causes a stack trace disconnect.
|
let span;
|
||||||
// We reconnect it by assigning the result of that dispatch to `opPromise`,
|
try {
|
||||||
// awaiting `opPromise` in an inner function also named `fetch()` and
|
if (internals.telemetry?.tracingEnabled) {
|
||||||
// returning the result from that.
|
span = new internals.telemetry.Span("fetch", { kind: 2 });
|
||||||
let opPromise = undefined;
|
internals.telemetry.enterSpan(span);
|
||||||
// 1.
|
|
||||||
const result = new Promise((resolve, reject) => {
|
|
||||||
const prefix = "Failed to execute 'fetch'";
|
|
||||||
webidl.requiredArguments(arguments.length, 1, prefix);
|
|
||||||
// 2.
|
|
||||||
const requestObject = new Request(input, init);
|
|
||||||
// 3.
|
|
||||||
const request = toInnerRequest(requestObject);
|
|
||||||
// 4.
|
|
||||||
if (requestObject.signal.aborted) {
|
|
||||||
reject(abortFetch(request, null, requestObject.signal.reason));
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 7.
|
// There is an async dispatch later that causes a stack trace disconnect.
|
||||||
let responseObject = null;
|
// We reconnect it by assigning the result of that dispatch to `opPromise`,
|
||||||
// 9.
|
// awaiting `opPromise` in an inner function also named `fetch()` and
|
||||||
let locallyAborted = false;
|
// returning the result from that.
|
||||||
// 10.
|
let opPromise = undefined;
|
||||||
function onabort() {
|
// 1.
|
||||||
locallyAborted = true;
|
const result = new Promise((resolve, reject) => {
|
||||||
reject(
|
const prefix = "Failed to execute 'fetch'";
|
||||||
abortFetch(request, responseObject, requestObject.signal.reason),
|
webidl.requiredArguments(arguments.length, 1, prefix);
|
||||||
);
|
// 2.
|
||||||
}
|
const requestObject = new Request(input, init);
|
||||||
requestObject.signal[abortSignal.add](onabort);
|
|
||||||
|
|
||||||
if (!requestObject.headers.has("Accept")) {
|
if (span) {
|
||||||
ArrayPrototypePush(request.headerList, ["Accept", "*/*"]);
|
span.updateName(requestObject.method);
|
||||||
}
|
span.setAttribute("http.request.method", requestObject.method);
|
||||||
|
const url = new URL(requestObject.url);
|
||||||
|
span.setAttribute("url.full", requestObject.url);
|
||||||
|
span.setAttribute(
|
||||||
|
"url.scheme",
|
||||||
|
StringPrototypeSlice(url.protocol, 0, -1),
|
||||||
|
);
|
||||||
|
span.setAttribute("url.path", url.pathname);
|
||||||
|
span.setAttribute("url.query", StringPrototypeSlice(url.search, 1));
|
||||||
|
}
|
||||||
|
|
||||||
if (!requestObject.headers.has("Accept-Language")) {
|
// 3.
|
||||||
ArrayPrototypePush(request.headerList, ["Accept-Language", "*"]);
|
const request = toInnerRequest(requestObject);
|
||||||
}
|
// 4.
|
||||||
|
if (requestObject.signal.aborted) {
|
||||||
|
reject(abortFetch(request, null, requestObject.signal.reason));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 7.
|
||||||
|
let responseObject = null;
|
||||||
|
// 9.
|
||||||
|
let locallyAborted = false;
|
||||||
|
// 10.
|
||||||
|
function onabort() {
|
||||||
|
locallyAborted = true;
|
||||||
|
reject(
|
||||||
|
abortFetch(request, responseObject, requestObject.signal.reason),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
requestObject.signal[abortSignal.add](onabort);
|
||||||
|
|
||||||
// 12.
|
if (!requestObject.headers.has("Accept")) {
|
||||||
opPromise = PromisePrototypeCatch(
|
ArrayPrototypePush(request.headerList, ["Accept", "*/*"]);
|
||||||
PromisePrototypeThen(
|
}
|
||||||
mainFetch(request, false, requestObject.signal),
|
|
||||||
(response) => {
|
if (!requestObject.headers.has("Accept-Language")) {
|
||||||
// 12.1.
|
ArrayPrototypePush(request.headerList, ["Accept-Language", "*"]);
|
||||||
if (locallyAborted) return;
|
}
|
||||||
// 12.2.
|
|
||||||
if (response.aborted) {
|
// 12.
|
||||||
reject(
|
opPromise = PromisePrototypeCatch(
|
||||||
abortFetch(
|
PromisePrototypeThen(
|
||||||
request,
|
mainFetch(request, false, requestObject.signal),
|
||||||
responseObject,
|
(response) => {
|
||||||
requestObject.signal.reason,
|
// 12.1.
|
||||||
),
|
if (locallyAborted) return;
|
||||||
);
|
// 12.2.
|
||||||
|
if (response.aborted) {
|
||||||
|
reject(
|
||||||
|
abortFetch(
|
||||||
|
request,
|
||||||
|
responseObject,
|
||||||
|
requestObject.signal.reason,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
requestObject.signal[abortSignal.remove](onabort);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// 12.3.
|
||||||
|
if (response.type === "error") {
|
||||||
|
const err = new TypeError(
|
||||||
|
"Fetch failed: " + (response.error ?? "unknown error"),
|
||||||
|
);
|
||||||
|
reject(err);
|
||||||
|
requestObject.signal[abortSignal.remove](onabort);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
responseObject = fromInnerResponse(response, "immutable");
|
||||||
|
|
||||||
|
if (span) {
|
||||||
|
span.setAttribute(
|
||||||
|
"http.response.status_code",
|
||||||
|
String(responseObject.status),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
resolve(responseObject);
|
||||||
requestObject.signal[abortSignal.remove](onabort);
|
requestObject.signal[abortSignal.remove](onabort);
|
||||||
return;
|
},
|
||||||
}
|
),
|
||||||
// 12.3.
|
(err) => {
|
||||||
if (response.type === "error") {
|
reject(err);
|
||||||
const err = new TypeError(
|
|
||||||
"Fetch failed: " + (response.error ?? "unknown error"),
|
|
||||||
);
|
|
||||||
reject(err);
|
|
||||||
requestObject.signal[abortSignal.remove](onabort);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
responseObject = fromInnerResponse(response, "immutable");
|
|
||||||
resolve(responseObject);
|
|
||||||
requestObject.signal[abortSignal.remove](onabort);
|
requestObject.signal[abortSignal.remove](onabort);
|
||||||
},
|
},
|
||||||
),
|
);
|
||||||
(err) => {
|
});
|
||||||
reject(err);
|
|
||||||
requestObject.signal[abortSignal.remove](onabort);
|
if (opPromise) {
|
||||||
},
|
PromisePrototypeCatch(result, () => {});
|
||||||
);
|
return (async function fetch() {
|
||||||
});
|
try {
|
||||||
if (opPromise) {
|
await opPromise;
|
||||||
PromisePrototypeCatch(result, () => {});
|
return result;
|
||||||
return (async function fetch() {
|
} finally {
|
||||||
await opPromise;
|
if (span) {
|
||||||
return result;
|
internals.telemetry.endSpan(span);
|
||||||
})();
|
}
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
// We need to end the span when the promise settles.
|
||||||
|
// WPT has a test that aborted fetch is settled in the same tick.
|
||||||
|
// This means we cannot wrap the promise if it is already settled.
|
||||||
|
// But this is OK, because we can just immediately end the span
|
||||||
|
// in that case.
|
||||||
|
if (span) {
|
||||||
|
// XXX: This should always be true, otherwise `opPromise` would be present.
|
||||||
|
if (op_fetch_promise_is_settled(result)) {
|
||||||
|
// It's already settled.
|
||||||
|
internals.telemetry.endSpan(span);
|
||||||
|
} else {
|
||||||
|
// Not settled yet, we can return a new wrapper promise.
|
||||||
|
return SafePromisePrototypeFinally(result, () => {
|
||||||
|
internals.telemetry.endSpan(span);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
} finally {
|
||||||
|
if (span) {
|
||||||
|
internals.telemetry.exitSpan(span);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function abortFetch(request, responseObject, error) {
|
function abortFetch(request, responseObject, error) {
|
||||||
|
|
|
@ -27,6 +27,7 @@ use deno_core::futures::TryFutureExt;
|
||||||
use deno_core::op2;
|
use deno_core::op2;
|
||||||
use deno_core::url;
|
use deno_core::url;
|
||||||
use deno_core::url::Url;
|
use deno_core::url::Url;
|
||||||
|
use deno_core::v8;
|
||||||
use deno_core::AsyncRefCell;
|
use deno_core::AsyncRefCell;
|
||||||
use deno_core::AsyncResult;
|
use deno_core::AsyncResult;
|
||||||
use deno_core::BufView;
|
use deno_core::BufView;
|
||||||
|
@ -141,6 +142,7 @@ deno_core::extension!(deno_fetch,
|
||||||
op_fetch_send,
|
op_fetch_send,
|
||||||
op_utf8_to_byte_string,
|
op_utf8_to_byte_string,
|
||||||
op_fetch_custom_client<FP>,
|
op_fetch_custom_client<FP>,
|
||||||
|
op_fetch_promise_is_settled,
|
||||||
],
|
],
|
||||||
esm = [
|
esm = [
|
||||||
"20_headers.js",
|
"20_headers.js",
|
||||||
|
@ -1206,3 +1208,8 @@ pub fn extract_authority(url: &mut Url) -> Option<(String, Option<String>)> {
|
||||||
|
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_fetch_promise_is_settled(promise: v8::Local<v8::Promise>) -> bool {
|
||||||
|
promise.state() != v8::PromiseState::Pending
|
||||||
|
}
|
||||||
|
|
|
@ -617,13 +617,13 @@ function mapToCallback(context, callback, onError) {
|
||||||
fastSyncResponseOrStream(req, inner.body, status, innerRequest);
|
fastSyncResponseOrStream(req, inner.body, status, innerRequest);
|
||||||
};
|
};
|
||||||
|
|
||||||
if (internals.telemetry.tracingEnabled) {
|
if (internals.telemetry?.tracingEnabled) {
|
||||||
const { Span, enterSpan, endSpan } = internals.telemetry;
|
const { Span, enterSpan, endSpan } = internals.telemetry;
|
||||||
const origMapped = mapped;
|
const origMapped = mapped;
|
||||||
mapped = function (req, _span) {
|
mapped = function (req, _span) {
|
||||||
const oldCtx = getAsyncContext();
|
const oldCtx = getAsyncContext();
|
||||||
setAsyncContext(context.asyncContext);
|
setAsyncContext(context.asyncContext);
|
||||||
const span = new Span("deno.serve");
|
const span = new Span("deno.serve", { kind: 1 });
|
||||||
try {
|
try {
|
||||||
enterSpan(span);
|
enterSpan(span);
|
||||||
return SafePromisePrototypeFinally(
|
return SafePromisePrototypeFinally(
|
||||||
|
|
|
@ -41,6 +41,8 @@ const { AsyncVariable, setAsyncContext } = core;
|
||||||
let TRACING_ENABLED = false;
|
let TRACING_ENABLED = false;
|
||||||
let DETERMINISTIC = false;
|
let DETERMINISTIC = false;
|
||||||
|
|
||||||
|
// Note: These start at 0 in the JS library,
|
||||||
|
// but start at 1 when serialized with JSON.
|
||||||
enum SpanKind {
|
enum SpanKind {
|
||||||
INTERNAL = 0,
|
INTERNAL = 0,
|
||||||
SERVER = 1,
|
SERVER = 1,
|
||||||
|
@ -91,6 +93,11 @@ interface Attributes {
|
||||||
|
|
||||||
type SpanAttributes = Attributes;
|
type SpanAttributes = Attributes;
|
||||||
|
|
||||||
|
interface SpanOptions {
|
||||||
|
attributes?: Attributes;
|
||||||
|
kind?: SpanKind;
|
||||||
|
}
|
||||||
|
|
||||||
interface Link {
|
interface Link {
|
||||||
context: SpanContext;
|
context: SpanContext;
|
||||||
attributes?: SpanAttributes;
|
attributes?: SpanAttributes;
|
||||||
|
@ -354,7 +361,7 @@ export class Span {
|
||||||
|
|
||||||
#recording = TRACING_ENABLED;
|
#recording = TRACING_ENABLED;
|
||||||
|
|
||||||
#kind: number = 0;
|
#kind: number = SpanKind.INTERNAL;
|
||||||
#name: string;
|
#name: string;
|
||||||
#startTime: number;
|
#startTime: number;
|
||||||
#status: { code: number; message?: string } | null = null;
|
#status: { code: number; message?: string } | null = null;
|
||||||
|
@ -429,7 +436,7 @@ export class Span {
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
name: string,
|
name: string,
|
||||||
attributes?: Attributes,
|
options?: SpanOptions,
|
||||||
) {
|
) {
|
||||||
if (!this.isRecording) {
|
if (!this.isRecording) {
|
||||||
this.#name = "";
|
this.#name = "";
|
||||||
|
@ -442,7 +449,8 @@ export class Span {
|
||||||
|
|
||||||
this.#name = name;
|
this.#name = name;
|
||||||
this.#startTime = now();
|
this.#startTime = now();
|
||||||
this.#attributes = attributes ?? { __proto__: null } as never;
|
this.#attributes = options?.attributes ?? { __proto__: null } as never;
|
||||||
|
this.#kind = options?.kind ?? SpanKind.INTERNAL;
|
||||||
|
|
||||||
const currentSpan: Span | {
|
const currentSpan: Span | {
|
||||||
spanContext(): { traceId: string; spanId: string };
|
spanContext(): { traceId: string; spanId: string };
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
"parentSpanId": "",
|
"parentSpanId": "",
|
||||||
"flags": 1,
|
"flags": 1,
|
||||||
"name": "GET",
|
"name": "GET",
|
||||||
"kind": 1,
|
"kind": 3,
|
||||||
"startTimeUnixNano": "[WILDCARD]",
|
"startTimeUnixNano": "[WILDCARD]",
|
||||||
"endTimeUnixNano": "[WILDCARD]",
|
"endTimeUnixNano": "[WILDCARD]",
|
||||||
"attributes": [
|
"attributes": [
|
||||||
|
@ -59,10 +59,68 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"traceId": "00000000000000000000000000000001",
|
"traceId": "00000000000000000000000000000003",
|
||||||
|
"spanId": "0000000000000004",
|
||||||
|
"traceState": "",
|
||||||
|
"parentSpanId": "",
|
||||||
|
"flags": 1,
|
||||||
|
"name": "GET",
|
||||||
|
"kind": 2,
|
||||||
|
"startTimeUnixNano": "[WILDCARD]",
|
||||||
|
"endTimeUnixNano": "[WILDCARD]",
|
||||||
|
"attributes": [
|
||||||
|
{
|
||||||
|
"key": "http.request.method",
|
||||||
|
"value": {
|
||||||
|
"stringValue": "GET"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"key": "url.full",
|
||||||
|
"value": {
|
||||||
|
"stringValue": "http://localhost:[WILDCARD]/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"key": "url.scheme",
|
||||||
|
"value": {
|
||||||
|
"stringValue": "http"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"key": "url.path",
|
||||||
|
"value": {
|
||||||
|
"stringValue": "/"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"key": "url.query",
|
||||||
|
"value": {
|
||||||
|
"stringValue": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"key": "http.response.status_code",
|
||||||
|
"value": {
|
||||||
|
"stringValue": "200"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"droppedAttributesCount": 0,
|
||||||
|
"events": [],
|
||||||
|
"droppedEventsCount": 0,
|
||||||
|
"links": [],
|
||||||
|
"droppedLinksCount": 0,
|
||||||
|
"status": {
|
||||||
|
"message": "",
|
||||||
|
"code": 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"traceId": "00000000000000000000000000000003",
|
||||||
"spanId": "1000000000000001",
|
"spanId": "1000000000000001",
|
||||||
"traceState": "",
|
"traceState": "",
|
||||||
"parentSpanId": "0000000000000002",
|
"parentSpanId": "0000000000000004",
|
||||||
"flags": 1,
|
"flags": 1,
|
||||||
"name": "outer span",
|
"name": "outer span",
|
||||||
"kind": 1,
|
"kind": 1,
|
||||||
|
@ -80,7 +138,7 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"traceId": "00000000000000000000000000000001",
|
"traceId": "00000000000000000000000000000003",
|
||||||
"spanId": "1000000000000002",
|
"spanId": "1000000000000002",
|
||||||
"traceState": "",
|
"traceState": "",
|
||||||
"parentSpanId": "1000000000000001",
|
"parentSpanId": "1000000000000001",
|
||||||
|
@ -113,7 +171,7 @@
|
||||||
"attributes": [],
|
"attributes": [],
|
||||||
"droppedAttributesCount": 0,
|
"droppedAttributesCount": 0,
|
||||||
"flags": 1,
|
"flags": 1,
|
||||||
"traceId": "00000000000000000000000000000001",
|
"traceId": "00000000000000000000000000000003",
|
||||||
"spanId": "1000000000000002"
|
"spanId": "1000000000000002"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -127,7 +185,7 @@
|
||||||
"attributes": [],
|
"attributes": [],
|
||||||
"droppedAttributesCount": 0,
|
"droppedAttributesCount": 0,
|
||||||
"flags": 1,
|
"flags": 1,
|
||||||
"traceId": "00000000000000000000000000000001",
|
"traceId": "00000000000000000000000000000003",
|
||||||
"spanId": "1000000000000002"
|
"spanId": "1000000000000002"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
Loading…
Add table
Reference in a new issue