You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tobspr_shapez.io/src/js/platform/api.js

213 lines
5.9 KiB

/* typehints:start */
import { Application } from "../application";
/* typehints:end */
import { createLogger } from "../core/logging";
import { compressX64 } from "../core/lzstring";
import { getIPCRenderer } from "../core/utils";
import { T } from "../translations";
const logger = createLogger("puzzle-api");
export class ClientAPI {
/**
*
* @param {Application} app
*/
constructor(app) {
this.app = app;
/**
* The current users session token
* @type {string|null}
*/
this.token = null;
}
getEndpoint() {
if (G_IS_DEV) {
return "http://localhost:15001";
}
if (window.location.host === "beta.shapez.io") {
return "https://api-staging.shapez.io";
}
return "https://api.shapez.io";
}
isLoggedIn() {
return Boolean(this.token);
}
/**
*
* @param {string} endpoint
* @param {object} options
* @param {"GET"|"POST"=} options.method
* @param {any=} options.body
*/
_request(endpoint, options) {
const headers = {
"x-api-key": "d5c54aaa491f200709afff082c153ef2",
"Content-Type": "application/json",
};
if (this.token) {
headers["x-token"] = this.token;
}
return Promise.race([
fetch(this.getEndpoint() + endpoint, {
cache: "no-cache",
mode: "cors",
headers,
method: options.method || "GET",
body: options.body ? JSON.stringify(options.body) : undefined,
})
.then(res => {
if (res.status !== 200) {
throw "bad-status: " + res.status + " / " + res.statusText;
}
return res;
})
.then(res => res.json()),
new Promise((resolve, reject) => setTimeout(() => reject("timeout"), 15000)),
])
.then(data => {
if (data && data.error) {
logger.warn("Got error from api:", data);
throw T.backendErrors[data.error] || data.error;
}
return data;
})
.catch(err => {
logger.warn("Failure:", endpoint, ":", err);
throw err;
});
}
tryLogin() {
return this.apiTryLogin()
.then(({ token }) => {
this.token = token;
return true;
})
.catch(err => {
logger.warn("Failed to login:", err);
return false;
});
}
/**
* @returns {Promise<{token: string}>}
*/
apiTryLogin() {
if (!G_IS_STANDALONE) {
const token = window.prompt(
"Please enter the auth token for the puzzle DLC (If you have none, you can't login):"
);
return Promise.resolve({ token });
}
const renderer = getIPCRenderer();
return renderer.invoke("steam:get-ticket").then(
ticket => {
logger.log("Got auth ticket:", ticket);
return this._request("/v1/public/login", {
method: "POST",
body: {
token: ticket,
},
});
},
err => {
logger.error("Failed to get auth ticket from steam: ", err);
throw err;
}
);
}
/**
* @param {"new"|"top-rated"|"mine"} category
* @returns {Promise<import("../savegame/savegame_typedefs").PuzzleMetadata[]>}
*/
apiListPuzzles(category) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/list/" + category, {});
}
/**
* @param {number} puzzleId
* @returns {Promise<import("../savegame/savegame_typedefs").PuzzleFullData>}
*/
apiDownloadPuzzle(puzzleId) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/download/" + puzzleId, {});
}
/**
* @param {number} shortKey
* @returns {Promise<import("../savegame/savegame_typedefs").PuzzleFullData>}
*/
apiDownloadPuzzleByKey(shortKey) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/download/" + shortKey, {});
}
/**
* @param {number} puzzleId
* @returns {Promise<void>}
*/
apiReportPuzzle(puzzleId, reason) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/report/" + puzzleId, {
method: "POST",
body: { reason },
});
}
/**
* @param {number} puzzleId
* @param {object} payload
* @param {number} payload.time
* @param {boolean} payload.liked
* @returns {Promise<{ success: true }>}
*/
apiCompletePuzzle(puzzleId, payload) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/complete/" + puzzleId, {
method: "POST",
body: payload,
});
}
/**
* @param {object} payload
* @param {string} payload.title
* @param {string} payload.shortKey
* @param {import("../savegame/savegame_typedefs").PuzzleGameData} payload.data
* @returns {Promise<{ success: true }>}
*/
apiSubmitPuzzle(payload) {
if (!this.isLoggedIn()) {
return Promise.reject("not-logged-in");
}
return this._request("/v1/puzzles/submit", {
method: "POST",
body: {
...payload,
data: compressX64(JSON.stringify(payload.data)),
},
});
}
}