budibase/packages/server/src/integrations/rest.ts

236 lines
5.5 KiB
TypeScript
Raw Normal View History

2021-06-24 19:17:26 +02:00
import {
Integration,
DatasourceFieldTypes,
QueryTypes,
} from "../definitions/datasource"
import { IntegrationBase } from "./base/IntegrationBase"
2021-12-02 18:53:14 +01:00
const BodyTypes = {
NONE: "none",
FORM_DATA: "form",
ENCODED: "encoded",
JSON: "json",
TEXT: "text",
}
enum AuthType {
BASIC = "basic",
BEARER = "bearer"
}
interface AuthConfig {
id: string
type: AuthType
config: BasicAuthConfig | BearerAuthConfig
}
2021-12-08 11:52:08 +01:00
interface BasicAuthConfig {
username: string,
password: string,
}
interface BearerAuthConfig {
token: string,
}
2021-12-02 18:53:14 +01:00
const coreFields = {
path: {
type: DatasourceFieldTypes.STRING,
display: "URL",
},
queryString: {
type: DatasourceFieldTypes.STRING,
},
headers: {
type: DatasourceFieldTypes.OBJECT,
},
enabledHeaders: {
type: DatasourceFieldTypes.OBJECT,
},
requestBody: {
type: DatasourceFieldTypes.JSON,
},
bodyType: {
type: DatasourceFieldTypes.STRING,
enum: Object.values(BodyTypes),
2021-12-02 18:53:14 +01:00
},
}
module RestModule {
const fetch = require("node-fetch")
const { formatBytes } = require("../utilities")
const { performance } = require("perf_hooks")
interface RestConfig {
2021-06-24 19:17:26 +02:00
url: string
defaultHeaders: {
2021-06-24 19:17:26 +02:00
[key: string]: any
}
authConfigs: AuthConfig[]
}
interface Request {
path: string
queryString?: string
headers?: string
json?: any
}
const SCHEMA: Integration = {
docs: "https://github.com/node-fetch/node-fetch",
description:
2021-11-30 19:11:29 +01:00
"With the REST API datasource, you can connect, query and pull data from multiple REST APIs. You can then use the retrieved data to build apps.",
friendlyName: "REST API",
datasource: {
url: {
type: DatasourceFieldTypes.STRING,
default: "",
required: false,
deprecated: true,
},
defaultHeaders: {
type: DatasourceFieldTypes.OBJECT,
required: false,
default: {},
},
},
query: {
create: {
readable: true,
displayName: "POST",
type: QueryTypes.FIELDS,
2021-12-02 18:53:14 +01:00
fields: coreFields,
},
read: {
displayName: "GET",
readable: true,
type: QueryTypes.FIELDS,
2021-12-02 18:53:14 +01:00
fields: coreFields,
},
update: {
displayName: "PUT",
readable: true,
type: QueryTypes.FIELDS,
2021-12-02 18:53:14 +01:00
fields: coreFields,
},
patch: {
displayName: "PATCH",
readable: true,
type: QueryTypes.FIELDS,
2021-12-02 18:53:14 +01:00
fields: coreFields,
},
delete: {
displayName: "DELETE",
type: QueryTypes.FIELDS,
2021-12-02 18:53:14 +01:00
fields: coreFields,
},
},
}
class RestIntegration implements IntegrationBase {
private config: RestConfig
private headers: {
[key: string]: string
} = {}
private startTimeMs: number = performance.now()
constructor(config: RestConfig) {
this.config = config
}
async parseResponse(response: any) {
2021-12-06 19:35:44 +01:00
let data, raw
const contentType = response.headers.get("content-type")
if (contentType && contentType.indexOf("application/json") !== -1) {
data = await response.json()
2021-12-06 19:35:44 +01:00
raw = JSON.stringify(data)
} else {
data = await response.text()
2021-12-06 19:35:44 +01:00
raw = data
}
const size = formatBytes(response.headers.get("content-length") || 0)
const time = `${Math.round(performance.now() - this.startTimeMs)}ms`
return {
data,
info: {
code: response.status,
size,
time,
},
2021-12-06 19:35:44 +01:00
raw,
}
}
getUrl(path: string, queryString: string): string {
const main = `${path}?${queryString}`
if (!this.config.url) {
return main
} else {
return `${this.config.url}/${main}`
}
}
processAuth(authConfigId: string) {
2021-12-08 11:52:08 +01:00
if (!this.config.authConfigs || !authConfigId) {
return
}
const authConfig = this.config.authConfigs.filter(authConfig => authConfig.id === authConfigId)[0]
let config
switch (authConfig.type) {
case AuthType.BASIC:
config = authConfig.config as BasicAuthConfig
this.headers.Authorization = `Basic ${Buffer.from(`${config.username}:${config.password}`).toString("base64")}`
break
case AuthType.BEARER:
config = authConfig.config as BearerAuthConfig
this.headers.Authorization = `Bearer ${config.token}`
break
}
}
async _req({ path = "", queryString = "", headers = {}, json = {}, method = "GET", authConfigId = "" }) {
this.headers = {
...this.config.defaultHeaders,
...headers,
}
this.processAuth(authConfigId)
const input: any = { method, headers: this.headers }
if (json && typeof json === "object" && Object.keys(json).length > 0) {
input.body = JSON.stringify(json)
}
this.startTimeMs = performance.now()
const response = await fetch(this.getUrl(path, queryString), input)
return await this.parseResponse(response)
}
async create(opts: Request) {
return this._req({ ...opts, method: "POST" })
}
async read(opts: Request) {
return this._req({ ...opts, method: "GET" })
}
async update(opts: Request) {
return this._req({ ...opts, method: "PUT" })
}
async patch(opts: Request) {
return this._req({ ...opts, method: "PATCH" })
}
async delete(opts: Request) {
return this._req({ ...opts, method: "DELETE" })
}
}
module.exports = {
schema: SCHEMA,
integration: RestIntegration,
AuthType
}
}