mirror of
https://github.com/alangrainger/immich-public-proxy.git
synced 2025-01-19 14:16:45 +01:00
120 lines
No EOL
13 KiB
JavaScript
120 lines
No EOL
13 KiB
JavaScript
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const tslib_1 = require("tslib");
|
|
const types_1 = require("./types");
|
|
const dayjs_1 = tslib_1.__importDefault(require("dayjs"));
|
|
const index_1 = require("./index");
|
|
class Immich {
|
|
/**
|
|
* Make a request to Immich API. We're not using the SDK to limit
|
|
* the possible attack surface of this app.
|
|
*/
|
|
request(endpoint) {
|
|
return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
|
const res = yield fetch(process.env.IMMICH_URL + '/api' + endpoint, {
|
|
headers: {
|
|
'x-api-key': process.env.API_KEY || ''
|
|
}
|
|
});
|
|
if (res.status === 200) {
|
|
const contentType = res.headers.get('Content-Type') || '';
|
|
if (contentType.includes('application/json')) {
|
|
return res.json();
|
|
}
|
|
else {
|
|
return res;
|
|
}
|
|
}
|
|
else {
|
|
(0, index_1.log)('Immich API status ' + res.status);
|
|
console.log(yield res.text());
|
|
}
|
|
});
|
|
}
|
|
/**
|
|
* Query Immich for the SharedLink metadata for a given key.
|
|
* The key is what is returned in the URL when you create a share in Immich.
|
|
*
|
|
* Immich doesn't have a method to query by key, so this method gets all
|
|
* known shared links, and returns the link which matches the provided key.
|
|
*/
|
|
getShareByKey(key) {
|
|
return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
|
var _a;
|
|
const res = ((yield this.request('/shared-links')) || []);
|
|
const link = res.find(x => x.key === key);
|
|
if (link) {
|
|
if (link.expiresAt && (0, dayjs_1.default)(link.expiresAt) < (0, dayjs_1.default)()) {
|
|
// This link has expired
|
|
(0, index_1.log)('Expired link ' + key);
|
|
}
|
|
else {
|
|
if (link.type === 'ALBUM') {
|
|
// Fetch the assets from the album and populate the SharedLink assets array
|
|
const albumId = (_a = link.album) === null || _a === void 0 ? void 0 : _a.id;
|
|
if (albumId) {
|
|
const album = (yield this.request('/albums/' + encodeURIComponent(albumId)));
|
|
link.assets = album.assets || [];
|
|
}
|
|
}
|
|
// Filter assets to exclude trashed assets
|
|
link.assets = link.assets.filter(x => !x.isTrashed);
|
|
return link;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
/**
|
|
* Stream asset buffer data from Immich.
|
|
*
|
|
* For photos, you can request 'thumbnail' or 'original' size.
|
|
* For videos, it is Immich's streaming quality, not the original quality.
|
|
*/
|
|
getAssetBuffer(asset, size) {
|
|
return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
|
switch (asset.type) {
|
|
case types_1.AssetType.image:
|
|
size = size === types_1.ImageSize.thumbnail ? types_1.ImageSize.thumbnail : types_1.ImageSize.original;
|
|
return this.request('/assets/' + encodeURIComponent(asset.id) + '/' + size);
|
|
case types_1.AssetType.video:
|
|
return this.request('/assets/' + encodeURIComponent(asset.id) + '/video/playback');
|
|
}
|
|
});
|
|
}
|
|
/**
|
|
* Get the content-type of an Immich asset
|
|
*/
|
|
getContentType(asset) {
|
|
return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
|
const assetBuffer = yield this.getAssetBuffer(asset);
|
|
return assetBuffer.headers.get('Content-Type');
|
|
});
|
|
}
|
|
/**
|
|
* Return the image data URL for a photo
|
|
*/
|
|
photoUrl(key, id, size) {
|
|
return `/photo/${key}/${id}` + (size ? `?size=${size}` : '');
|
|
}
|
|
/**
|
|
* Return the video data URL for a video
|
|
*/
|
|
videoUrl(key, id) {
|
|
return `/video/${key}/${id}`;
|
|
}
|
|
/**
|
|
* Check if a provided ID matches the Immich ID format
|
|
*/
|
|
isId(id) {
|
|
return !!id.match(/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/);
|
|
}
|
|
/**
|
|
* Check if a provided key matches the Immich shared-link key format
|
|
*/
|
|
isKey(key) {
|
|
return !!key.match(/^[\w-]+$/);
|
|
}
|
|
}
|
|
const immich = new Immich();
|
|
exports.default = immich;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"immich.js","sourceRoot":"","sources":["../src/immich.ts"],"names":[],"mappings":"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","sourcesContent":["import { Album, Asset, AssetType, ImageSize, SharedLink } from './types'\nimport dayjs from 'dayjs'\nimport { log } from './index'\n\nclass Immich {\n  /**\n   * Make a request to Immich API. We're not using the SDK to limit\n   * the possible attack surface of this app.\n   */\n  async request (endpoint: string) {\n    const res = await fetch(process.env.IMMICH_URL + '/api' + endpoint, {\n      headers: {\n        'x-api-key': process.env.API_KEY || ''\n      }\n    })\n    if (res.status === 200) {\n      const contentType = res.headers.get('Content-Type') || ''\n      if (contentType.includes('application/json')) {\n        return res.json()\n      } else {\n        return res\n      }\n    } else {\n      log('Immich API status ' + res.status)\n      console.log(await res.text())\n    }\n  }\n\n  /**\n   * Query Immich for the SharedLink metadata for a given key.\n   * The key is what is returned in the URL when you create a share in Immich.\n   *\n   * Immich doesn't have a method to query by key, so this method gets all\n   * known shared links, and returns the link which matches the provided key.\n   */\n  async getShareByKey (key: string) {\n    const res = (await this.request('/shared-links') || []) as SharedLink[]\n    const link = res.find(x => x.key === key)\n    if (link) {\n      if (link.expiresAt && dayjs(link.expiresAt) < dayjs()) {\n        // This link has expired\n        log('Expired link ' + key)\n      } else {\n        if (link.type === 'ALBUM') {\n          // Fetch the assets from the album and populate the SharedLink assets array\n          const albumId = link.album?.id\n          if (albumId) {\n            const album = (await this.request('/albums/' + encodeURIComponent(albumId))) as Album\n            link.assets = album.assets || []\n          }\n        }\n        // Filter assets to exclude trashed assets\n        link.assets = link.assets.filter(x => !x.isTrashed)\n        return link\n      }\n    }\n  }\n\n  /**\n   * Stream asset buffer data from Immich.\n   *\n   * For photos, you can request 'thumbnail' or 'original' size.\n   * For videos, it is Immich's streaming quality, not the original quality.\n   */\n  async getAssetBuffer (asset: Asset, size?: ImageSize) {\n    switch (asset.type) {\n      case AssetType.image:\n        size = size === ImageSize.thumbnail ? ImageSize.thumbnail : ImageSize.original\n        return this.request('/assets/' + encodeURIComponent(asset.id) + '/' + size)\n      case AssetType.video:\n        return this.request('/assets/' + encodeURIComponent(asset.id) + '/video/playback')\n    }\n  }\n\n  /**\n   * Get the content-type of an Immich asset\n   */\n  async getContentType (asset: Asset) {\n    const assetBuffer = await this.getAssetBuffer(asset)\n    return assetBuffer.headers.get('Content-Type')\n  }\n\n  /**\n   * Return the image data URL for a photo\n   */\n  photoUrl (key: string, id: string, size?: ImageSize) {\n    return `/photo/${key}/${id}` + (size ? `?size=${size}` : '')\n  }\n\n  /**\n   * Return the video data URL for a video\n   */\n  videoUrl (key: string, id: string) {\n    return `/video/${key}/${id}`\n  }\n\n  /**\n   * Check if a provided ID matches the Immich ID format\n   */\n  isId (id: string) {\n    return !!id.match(/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/)\n  }\n\n  /**\n   * Check if a provided key matches the Immich shared-link key format\n   */\n  isKey (key: string) {\n    return !!key.match(/^[\\w-]+$/)\n  }\n}\n\nconst immich = new Immich()\n\nexport default immich\n"]}
|