vector-im-hydrogen-web/src/matrix/hs-api.js

138 lines
4.6 KiB
JavaScript
Raw Normal View History

import {
2019-06-26 22:31:36 +02:00
HomeServerError,
RequestAbortError,
2019-03-08 12:26:59 +01:00
NetworkError
} from "./error.js";
2019-02-10 21:25:29 +01:00
2019-02-04 23:26:24 +01:00
class RequestWrapper {
2019-06-26 22:31:36 +02:00
constructor(promise, controller) {
if (!controller) {
const abortPromise = new Promise((_, reject) => {
this._controller = {
abort() {
const err = new Error("fetch request aborted");
err.name = "AbortError";
reject(err);
}
};
});
this._promise = Promise.race([promise, abortPromise]);
} else {
this._promise = promise;
this._controller = controller;
}
2019-06-26 22:31:36 +02:00
}
2018-12-21 14:35:24 +01:00
2019-06-26 22:31:36 +02:00
abort() {
this._controller.abort();
}
2018-12-21 14:35:24 +01:00
2019-06-26 22:31:36 +02:00
response() {
return this._promise;
}
2018-12-21 14:35:24 +01:00
}
2019-07-26 22:03:57 +02:00
// todo: everywhere here, encode params in the url that could have slashes ... mainly event ids?
2019-02-07 01:25:12 +01:00
export default class HomeServerApi {
2019-06-26 22:31:36 +02:00
constructor(homeserver, accessToken) {
2019-03-08 20:03:47 +01:00
// store these both in a closure somehow so it's harder to get at in case of XSS?
// one could change the homeserver as well so the token gets sent there, so both must be protected from read/write
this._homeserver = homeserver;
2019-06-26 22:31:36 +02:00
this._accessToken = accessToken;
}
2018-12-21 14:35:24 +01:00
2019-06-26 22:31:36 +02:00
_url(csPath) {
return `${this._homeserver}/_matrix/client/r0${csPath}`;
}
2018-12-21 14:35:24 +01:00
2019-06-26 22:31:36 +02:00
_request(method, csPath, queryParams = {}, body) {
const queryString = Object.entries(queryParams)
.filter(([, value]) => value !== undefined)
.map(([name, value]) => `${encodeURIComponent(name)}=${encodeURIComponent(value)}`)
.join("&");
const url = this._url(`${csPath}?${queryString}`);
let bodyString;
const headers = new Headers();
if (this._accessToken) {
headers.append("Authorization", `Bearer ${this._accessToken}`);
}
headers.append("Accept", "application/json");
if (body) {
headers.append("Content-Type", "application/json");
bodyString = JSON.stringify(body);
}
const controller = typeof AbortController === "function" ? new AbortController() : null;
// TODO: set authenticated headers with second arguments, cache them
let promise = fetch(url, {
method,
headers,
body: bodyString,
2019-09-15 12:23:08 +02:00
signal: controller && controller.signal,
mode: "cors",
credentials: "omit",
referrer: "no-referrer",
cache: "no-cache",
2019-06-26 22:31:36 +02:00
});
promise = promise.then(async (response) => {
if (response.ok) {
return await response.json();
} else {
switch (response.status) {
default:
throw new HomeServerError(method, url, await response.json())
}
}
}, err => {
2019-03-08 12:26:59 +01:00
if (err.name === "AbortError") {
throw new RequestAbortError();
} else if (err instanceof TypeError) {
// Network errors are reported as TypeErrors, see
// https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch#Checking_that_the_fetch_was_successful
// this can either mean user is offline, server is offline, or a CORS error (server misconfiguration).
//
// One could check navigator.onLine to rule out the first
// but the 2 later ones are indistinguishable from javascript.
throw new NetworkError(err.message);
}
2019-06-26 22:31:36 +02:00
throw err;
});
return new RequestWrapper(promise, controller);
}
2019-02-04 23:26:24 +01:00
2019-06-26 22:31:36 +02:00
_post(csPath, queryParams, body) {
return this._request("POST", csPath, queryParams, body);
}
2019-02-04 23:26:24 +01:00
2019-07-26 22:03:57 +02:00
_put(csPath, queryParams, body) {
return this._request("PUT", csPath, queryParams, body);
}
2019-06-26 22:31:36 +02:00
_get(csPath, queryParams, body) {
return this._request("GET", csPath, queryParams, body);
}
2018-12-21 14:35:24 +01:00
2019-06-26 22:31:36 +02:00
sync(since, filter, timeout) {
return this._get("/sync", {since, timeout, filter});
}
2019-02-04 23:26:24 +01:00
2019-03-09 00:41:06 +01:00
// params is from, dir and optionally to, limit, filter.
messages(roomId, params) {
return this._get(`/rooms/${roomId}/messages`, params);
}
2019-07-26 22:03:57 +02:00
send(roomId, eventType, txnId, content) {
return this._put(`/rooms/${roomId}/send/${eventType}/${txnId}`, {}, content);
}
2019-06-26 22:31:36 +02:00
passwordLogin(username, password) {
2019-02-04 23:26:24 +01:00
return this._post("/login", undefined, {
"type": "m.login.password",
"identifier": {
"type": "m.id.user",
"user": username
},
"password": password
});
2019-06-26 22:31:36 +02:00
}
2019-03-08 12:26:59 +01:00
}