2019-02-10 21:40:11 +01:00
|
|
|
import {
|
2019-06-26 22:31:36 +02:00
|
|
|
HomeServerError,
|
|
|
|
RequestAbortError,
|
2019-03-08 12:26:59 +01:00
|
|
|
NetworkError
|
2019-02-10 21:40:11 +01:00
|
|
|
} 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) {
|
2019-06-26 19:49:49 +02:00
|
|
|
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
|
|
|
}
|