* feat: add subscription manager to handle subscription sharing * fix(services): add missing methods to the interface * fix(services): add subscription manager specs * feat(snjs): add subscriptions e2e tests * fix(snjs): add wait in subscription cancelling test * fix(snjs): checking for canceled invitations in tests * fix(snjs): add e2e test for restored limit of subscription invitations * chore(lint): fix linter issues
213 lines
6.8 KiB
TypeScript
213 lines
6.8 KiB
TypeScript
import { API_MESSAGE_RATE_LIMITED, UNKNOWN_ERROR } from './Messages'
|
|
import { HttpResponse, StatusCode } from '@standardnotes/responses'
|
|
import { isString } from '@standardnotes/utils'
|
|
import { SnjsVersion } from '@Lib/Version'
|
|
import { AbstractService, InternalEventBusInterface } from '@standardnotes/services'
|
|
import { Environment } from '@standardnotes/models'
|
|
|
|
export enum HttpVerb {
|
|
Get = 'GET',
|
|
Post = 'POST',
|
|
Put = 'PUT',
|
|
Patch = 'PATCH',
|
|
Delete = 'DELETE',
|
|
}
|
|
|
|
const REQUEST_READY_STATE_COMPLETED = 4
|
|
|
|
export type HttpParams = Record<string, unknown>
|
|
|
|
export type HttpRequest = {
|
|
url: string
|
|
params?: HttpParams
|
|
rawBytes?: Uint8Array
|
|
verb: HttpVerb
|
|
authentication?: string
|
|
customHeaders?: Record<string, string>[]
|
|
responseType?: XMLHttpRequestResponseType
|
|
external?: boolean
|
|
}
|
|
|
|
/**
|
|
* A non-SNJS specific wrapper for XMLHttpRequests
|
|
*/
|
|
export class SNHttpService extends AbstractService {
|
|
constructor(
|
|
private readonly environment: Environment,
|
|
private readonly appVersion: string,
|
|
protected override internalEventBus: InternalEventBusInterface,
|
|
) {
|
|
super(internalEventBus)
|
|
}
|
|
|
|
public async getAbsolute(url: string, params?: HttpParams, authentication?: string): Promise<HttpResponse> {
|
|
return this.runHttp({ url, params, verb: HttpVerb.Get, authentication })
|
|
}
|
|
|
|
public async postAbsolute(url: string, params?: HttpParams, authentication?: string): Promise<HttpResponse> {
|
|
return this.runHttp({ url, params, verb: HttpVerb.Post, authentication })
|
|
}
|
|
|
|
public async putAbsolute(url: string, params?: HttpParams, authentication?: string): Promise<HttpResponse> {
|
|
return this.runHttp({ url, params, verb: HttpVerb.Put, authentication })
|
|
}
|
|
|
|
public async patchAbsolute(url: string, params: HttpParams, authentication?: string): Promise<HttpResponse> {
|
|
return this.runHttp({ url, params, verb: HttpVerb.Patch, authentication })
|
|
}
|
|
|
|
public async deleteAbsolute(url: string, params?: HttpParams, authentication?: string): Promise<HttpResponse> {
|
|
return this.runHttp({ url, params, verb: HttpVerb.Delete, authentication })
|
|
}
|
|
|
|
public async runHttp(httpRequest: HttpRequest): Promise<HttpResponse> {
|
|
const request = this.createXmlRequest(httpRequest)
|
|
|
|
return this.runRequest(request, this.createRequestBody(httpRequest))
|
|
}
|
|
|
|
private createRequestBody(httpRequest: HttpRequest): string | Uint8Array | undefined {
|
|
if (
|
|
httpRequest.params !== undefined &&
|
|
[HttpVerb.Post, HttpVerb.Put, HttpVerb.Patch, HttpVerb.Delete].includes(httpRequest.verb)
|
|
) {
|
|
return JSON.stringify(httpRequest.params)
|
|
}
|
|
|
|
return httpRequest.rawBytes
|
|
}
|
|
|
|
private createXmlRequest(httpRequest: HttpRequest) {
|
|
const request = new XMLHttpRequest()
|
|
if (httpRequest.params && httpRequest.verb === HttpVerb.Get && Object.keys(httpRequest.params).length > 0) {
|
|
httpRequest.url = this.urlForUrlAndParams(httpRequest.url, httpRequest.params)
|
|
}
|
|
request.open(httpRequest.verb, httpRequest.url, true)
|
|
request.responseType = httpRequest.responseType ?? ''
|
|
|
|
if (!httpRequest.external) {
|
|
request.setRequestHeader('X-SNJS-Version', SnjsVersion)
|
|
|
|
const appVersionHeaderValue = `${Environment[this.environment]}-${this.appVersion}`
|
|
request.setRequestHeader('X-Application-Version', appVersionHeaderValue)
|
|
|
|
if (httpRequest.authentication) {
|
|
request.setRequestHeader('Authorization', 'Bearer ' + httpRequest.authentication)
|
|
}
|
|
}
|
|
|
|
let contenTypeIsSet = false
|
|
if (httpRequest.customHeaders && httpRequest.customHeaders.length > 0) {
|
|
httpRequest.customHeaders.forEach(({ key, value }) => {
|
|
request.setRequestHeader(key, value)
|
|
if (key === 'Content-Type') {
|
|
contenTypeIsSet = true
|
|
}
|
|
})
|
|
}
|
|
if (!contenTypeIsSet && !httpRequest.external) {
|
|
request.setRequestHeader('Content-Type', 'application/json')
|
|
}
|
|
|
|
return request
|
|
}
|
|
|
|
private async runRequest(request: XMLHttpRequest, body?: string | Uint8Array): Promise<HttpResponse> {
|
|
return new Promise((resolve, reject) => {
|
|
request.onreadystatechange = () => {
|
|
this.stateChangeHandlerForRequest(request, resolve, reject)
|
|
}
|
|
request.send(body)
|
|
})
|
|
}
|
|
|
|
private stateChangeHandlerForRequest(
|
|
request: XMLHttpRequest,
|
|
resolve: (response: HttpResponse) => void,
|
|
reject: (response: HttpResponse) => void,
|
|
) {
|
|
if (request.readyState !== REQUEST_READY_STATE_COMPLETED) {
|
|
return
|
|
}
|
|
const httpStatus = request.status
|
|
const response: HttpResponse = {
|
|
status: httpStatus,
|
|
headers: new Map<string, string | null>(),
|
|
}
|
|
|
|
const responseHeaderLines = request
|
|
.getAllResponseHeaders()
|
|
?.trim()
|
|
.split(/[\r\n]+/)
|
|
responseHeaderLines?.forEach((responseHeaderLine) => {
|
|
const parts = responseHeaderLine.split(': ')
|
|
const name = parts.shift() as string
|
|
const value = parts.join(': ')
|
|
|
|
;(<Map<string, string | null>>response.headers).set(name, value)
|
|
})
|
|
|
|
try {
|
|
if (httpStatus !== StatusCode.HttpStatusNoContent) {
|
|
let body
|
|
|
|
const contentTypeHeader = response.headers?.get('content-type') || response.headers?.get('Content-Type')
|
|
|
|
if (contentTypeHeader?.includes('application/json')) {
|
|
body = JSON.parse(request.responseText)
|
|
} else {
|
|
body = request.response
|
|
}
|
|
/**
|
|
* v0 APIs do not have a `data` top-level object. In such cases, mimic
|
|
* the newer response body style by putting all the top-level
|
|
* properties inside a `data` object.
|
|
*/
|
|
if (!body.data) {
|
|
response.data = body
|
|
}
|
|
if (!isString(body)) {
|
|
Object.assign(response, body)
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.error(error)
|
|
}
|
|
if (httpStatus >= StatusCode.HttpStatusMinSuccess && httpStatus <= StatusCode.HttpStatusMaxSuccess) {
|
|
resolve(response)
|
|
} else {
|
|
if (httpStatus === StatusCode.HttpStatusForbidden) {
|
|
response.error = {
|
|
message: API_MESSAGE_RATE_LIMITED,
|
|
status: httpStatus,
|
|
}
|
|
} else if (response.error == undefined) {
|
|
if (response.data == undefined || response.data.error == undefined) {
|
|
try {
|
|
response.error = { message: request.responseText || UNKNOWN_ERROR, status: httpStatus }
|
|
} catch (error) {
|
|
response.error = { message: UNKNOWN_ERROR, status: httpStatus }
|
|
}
|
|
} else {
|
|
response.error = response.data.error
|
|
}
|
|
}
|
|
reject(response)
|
|
}
|
|
}
|
|
|
|
private urlForUrlAndParams(url: string, params: HttpParams) {
|
|
const keyValueString = Object.keys(params)
|
|
.map((key) => {
|
|
return key + '=' + encodeURIComponent(params[key] as string)
|
|
})
|
|
.join('&')
|
|
|
|
if (url.includes('?')) {
|
|
return url + '&' + keyValueString
|
|
} else {
|
|
return url + '?' + keyValueString
|
|
}
|
|
}
|
|
}
|