mirror of
https://github.com/tauri-apps/plugins-workspace.git
synced 2026-04-21 11:26:15 +02:00
61e9b0ab64
Co-authored-by: Amr Bashir <github@amrbashir.me>
302 lines
7.7 KiB
TypeScript
302 lines
7.7 KiB
TypeScript
// Copyright 2019-2023 Tauri Programme within The Commons Conservancy
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
/**
|
|
* Make HTTP requests with the Rust backend.
|
|
*
|
|
* ## Security
|
|
*
|
|
* This API has a scope configuration that forces you to restrict the URLs that can be accessed using glob patterns.
|
|
*
|
|
* For instance, this scope configuration only allows making HTTP requests to all subdomains for `tauri.app` except for `https://private.tauri.app`:
|
|
* ```json
|
|
* {
|
|
* "permissions": [
|
|
* {
|
|
* "identifier": "http:default",
|
|
* "allow": [{ "url": "https://*.tauri.app" }],
|
|
* "deny": [{ "url": "https://private.tauri.app" }]
|
|
* }
|
|
* ]
|
|
* }
|
|
* ```
|
|
* Trying to execute any API with a URL not configured on the scope results in a promise rejection due to denied access.
|
|
*
|
|
* @module
|
|
*/
|
|
|
|
import { invoke } from '@tauri-apps/api/core'
|
|
|
|
/**
|
|
* Configuration of a proxy that a Client should pass requests to.
|
|
*
|
|
* @since 2.0.0
|
|
*/
|
|
export interface Proxy {
|
|
/**
|
|
* Proxy all traffic to the passed URL.
|
|
*/
|
|
all?: string | ProxyConfig
|
|
/**
|
|
* Proxy all HTTP traffic to the passed URL.
|
|
*/
|
|
http?: string | ProxyConfig
|
|
/**
|
|
* Proxy all HTTPS traffic to the passed URL.
|
|
*/
|
|
https?: string | ProxyConfig
|
|
}
|
|
|
|
export interface ProxyConfig {
|
|
/**
|
|
* The URL of the proxy server.
|
|
*/
|
|
url: string
|
|
/**
|
|
* Set the `Proxy-Authorization` header using Basic auth.
|
|
*/
|
|
basicAuth?: {
|
|
username: string
|
|
password: string
|
|
}
|
|
/**
|
|
* A configuration for filtering out requests that shouldn't be proxied.
|
|
* Entries are expected to be comma-separated (whitespace between entries is ignored)
|
|
*/
|
|
noProxy?: string
|
|
}
|
|
|
|
/**
|
|
* Options to configure the Rust client used to make fetch requests
|
|
*
|
|
* @since 2.0.0
|
|
*/
|
|
export interface ClientOptions {
|
|
/**
|
|
* Defines the maximum number of redirects the client should follow.
|
|
* If set to 0, no redirects will be followed.
|
|
*/
|
|
maxRedirections?: number
|
|
/** Timeout in milliseconds */
|
|
connectTimeout?: number
|
|
/**
|
|
* Configuration of a proxy that a Client should pass requests to.
|
|
*/
|
|
proxy?: Proxy
|
|
/**
|
|
* Configuration for dangerous settings on the client such as disabling SSL verification.
|
|
*/
|
|
danger?: DangerousSettings
|
|
}
|
|
|
|
/**
|
|
* Configuration for dangerous settings on the client such as disabling SSL verification.
|
|
*
|
|
* @since 2.3.0
|
|
*/
|
|
export interface DangerousSettings {
|
|
/**
|
|
* Disables SSL verification.
|
|
*/
|
|
acceptInvalidCerts?: boolean
|
|
/**
|
|
* Disables hostname verification.
|
|
*/
|
|
acceptInvalidHostnames?: boolean
|
|
}
|
|
|
|
const ERROR_REQUEST_CANCELLED = 'Request cancelled'
|
|
|
|
/**
|
|
* Fetch a resource from the network. It returns a `Promise` that resolves to the
|
|
* `Response` to that `Request`, whether it is successful or not.
|
|
*
|
|
* @example
|
|
* ```typescript
|
|
* const response = await fetch("http://my.json.host/data.json");
|
|
* console.log(response.status); // e.g. 200
|
|
* console.log(response.statusText); // e.g. "OK"
|
|
* const jsonData = await response.json();
|
|
* ```
|
|
*
|
|
* @since 2.0.0
|
|
*/
|
|
export async function fetch(
|
|
input: URL | Request | string,
|
|
init?: RequestInit & ClientOptions
|
|
): Promise<Response> {
|
|
// Optimistically check for abort signal and avoid doing any work
|
|
const signal = init?.signal
|
|
if (signal?.aborted) {
|
|
throw new Error(ERROR_REQUEST_CANCELLED)
|
|
}
|
|
|
|
const maxRedirections = init?.maxRedirections
|
|
const connectTimeout = init?.connectTimeout
|
|
const proxy = init?.proxy
|
|
const danger = init?.danger
|
|
|
|
// Remove these fields before creating the request
|
|
if (init) {
|
|
delete init.maxRedirections
|
|
delete init.connectTimeout
|
|
delete init.proxy
|
|
delete init.danger
|
|
}
|
|
|
|
const headers = init?.headers
|
|
? init.headers instanceof Headers
|
|
? init.headers
|
|
: new Headers(init.headers)
|
|
: new Headers()
|
|
|
|
const req = new Request(input, init)
|
|
const buffer = await req.arrayBuffer()
|
|
const data =
|
|
buffer.byteLength !== 0 ? Array.from(new Uint8Array(buffer)) : null
|
|
|
|
// append new headers created by the browser `Request` implementation,
|
|
// if not already declared by the caller of this function
|
|
for (const [key, value] of req.headers) {
|
|
if (!headers.get(key)) {
|
|
headers.set(key, value)
|
|
}
|
|
}
|
|
|
|
const headersArray =
|
|
headers instanceof Headers
|
|
? Array.from(headers.entries())
|
|
: Array.isArray(headers)
|
|
? headers
|
|
: Object.entries(headers)
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
|
|
const mappedHeaders: Array<[string, string]> = headersArray.map(
|
|
([name, val]) => [
|
|
name,
|
|
// we need to ensure we have all header values as strings
|
|
// eslint-disable-next-line
|
|
typeof val === 'string' ? val : (val as any).toString()
|
|
]
|
|
)
|
|
|
|
// Optimistically check for abort signal and avoid doing any work on the Rust side
|
|
if (signal?.aborted) {
|
|
throw new Error(ERROR_REQUEST_CANCELLED)
|
|
}
|
|
|
|
const rid = await invoke<number>('plugin:http|fetch', {
|
|
clientConfig: {
|
|
method: req.method,
|
|
url: req.url,
|
|
headers: mappedHeaders,
|
|
data,
|
|
maxRedirections,
|
|
connectTimeout,
|
|
proxy,
|
|
danger
|
|
}
|
|
})
|
|
|
|
const abort = () => invoke('plugin:http|fetch_cancel', { rid })
|
|
|
|
// Optimistically check for abort signal
|
|
// and avoid doing any work after doing intial work on the Rust side
|
|
if (signal?.aborted) {
|
|
// we don't care about the result of this proimse
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
abort()
|
|
throw new Error(ERROR_REQUEST_CANCELLED)
|
|
}
|
|
|
|
signal?.addEventListener('abort', () => void abort())
|
|
|
|
interface FetchSendResponse {
|
|
status: number
|
|
statusText: string
|
|
headers: [[string, string]]
|
|
url: string
|
|
rid: number
|
|
}
|
|
|
|
const {
|
|
status,
|
|
statusText,
|
|
url,
|
|
headers: responseHeaders,
|
|
rid: responseRid
|
|
} = await invoke<FetchSendResponse>('plugin:http|fetch_send', {
|
|
rid
|
|
})
|
|
|
|
const dropBody = () => {
|
|
return invoke('plugin:http|fetch_cancel_body', { rid: responseRid })
|
|
}
|
|
|
|
const readChunk = async (
|
|
controller: ReadableStreamDefaultController<Uint8Array>
|
|
) => {
|
|
let data: ArrayBuffer
|
|
try {
|
|
data = await invoke('plugin:http|fetch_read_body', {
|
|
rid: responseRid
|
|
})
|
|
} catch (e) {
|
|
// close the stream if an error occurs
|
|
// and drop the body on Rust side
|
|
controller.error(e)
|
|
void dropBody()
|
|
return
|
|
}
|
|
|
|
const dataUint8 = new Uint8Array(data)
|
|
const lastByte = dataUint8[dataUint8.byteLength - 1]
|
|
const actualData = dataUint8.slice(0, dataUint8.byteLength - 1)
|
|
|
|
// close when the signal to close (last byte is 1) is sent from the IPC.
|
|
if (lastByte === 1) {
|
|
controller.close()
|
|
return
|
|
}
|
|
|
|
controller.enqueue(actualData)
|
|
}
|
|
|
|
// no body for 101, 103, 204, 205 and 304
|
|
// see https://fetch.spec.whatwg.org/#null-body-status
|
|
const body = [101, 103, 204, 205, 304].includes(status)
|
|
? null
|
|
: new ReadableStream<Uint8Array>({
|
|
start: (controller) => {
|
|
// listen for abort events to cancel reading
|
|
signal?.addEventListener('abort', () => {
|
|
controller.error(ERROR_REQUEST_CANCELLED)
|
|
void dropBody()
|
|
})
|
|
},
|
|
pull: (controller) => readChunk(controller),
|
|
cancel: () => {
|
|
// Ensure body resources are released on stream cancellation
|
|
void dropBody()
|
|
}
|
|
})
|
|
|
|
const res = new Response(body, {
|
|
status,
|
|
statusText
|
|
})
|
|
|
|
// Set `Response` properties that are ignored by the
|
|
// constructor, like url and some headers
|
|
//
|
|
// Since url and headers are read only properties
|
|
// this is the only way to set them.
|
|
Object.defineProperty(res, 'url', { value: url })
|
|
Object.defineProperty(res, 'headers', {
|
|
value: new Headers(responseHeaders)
|
|
})
|
|
|
|
return res
|
|
}
|