decap-cms-backend-proxy
Version:
Proxy backend for Decap CMS
292 lines • 6.6 kB
JavaScript
import { EditorialWorkflowError, APIError, unsentRequest, blobToFileObj } from 'decap-cms-lib-util';
import AuthenticationPage from './AuthenticationPage';
async function serializeAsset(assetProxy) {
const base64content = await assetProxy.toBase64();
return {
path: assetProxy.path,
content: base64content,
encoding: 'base64'
};
}
function deserializeMediaFile({
id,
content,
encoding,
path,
name
}) {
let byteArray = new Uint8Array(0);
if (encoding !== 'base64') {
console.error(`Unsupported encoding '${encoding}' for file '${path}'`);
} else {
const decodedContent = atob(content);
byteArray = new Uint8Array(decodedContent.length);
for (let i = 0; i < decodedContent.length; i++) {
byteArray[i] = decodedContent.charCodeAt(i);
}
}
const blob = new Blob([byteArray]);
const file = blobToFileObj(name, blob);
const url = URL.createObjectURL(file);
return {
id,
name,
path,
file,
size: file.size,
url,
displayURL: url
};
}
export default class ProxyBackend {
constructor(config, options = {}) {
if (!config.backend.proxy_url) {
throw new Error('The Proxy backend needs a "proxy_url" in the backend configuration.');
}
this.branch = config.backend.branch || 'master';
this.proxyUrl = config.backend.proxy_url;
this.mediaFolder = config.media_folder;
this.options = options;
this.cmsLabelPrefix = config.backend.cms_label_prefix;
}
isGitBackend() {
return false;
}
status() {
return Promise.resolve({
auth: {
status: true
},
api: {
status: true,
statusPage: ''
}
});
}
authComponent() {
return AuthenticationPage;
}
restoreUser() {
return this.authenticate();
}
authenticate() {
return Promise.resolve();
}
logout() {
return null;
}
getToken() {
return Promise.resolve('');
}
async request(payload) {
const response = await unsentRequest.fetchWithTimeout(this.proxyUrl, {
method: 'POST',
headers: {
'Content-Type': 'application/json; charset=utf-8'
},
body: JSON.stringify({
branch: this.branch,
...payload
})
});
const json = await response.json();
if (response.ok) {
return json;
} else {
throw new APIError(json.error, response.status, 'Proxy');
}
}
entriesByFolder(folder, extension, depth) {
return this.request({
action: 'entriesByFolder',
params: {
branch: this.branch,
folder,
extension,
depth
}
});
}
entriesByFiles(files) {
return this.request({
action: 'entriesByFiles',
params: {
branch: this.branch,
files
}
});
}
getEntry(path) {
return this.request({
action: 'getEntry',
params: {
branch: this.branch,
path
}
});
}
unpublishedEntries() {
return this.request({
action: 'unpublishedEntries',
params: {
branch: this.branch
}
});
}
async unpublishedEntry({
id,
collection,
slug
}) {
try {
const entry = await this.request({
action: 'unpublishedEntry',
params: {
branch: this.branch,
id,
collection,
slug,
cmsLabelPrefix: this.cmsLabelPrefix
}
});
return entry;
} catch (e) {
if (e.status === 404) {
throw new EditorialWorkflowError('content is not under editorial workflow', true);
}
throw e;
}
}
async unpublishedEntryDataFile(collection, slug, path, id) {
const {
data
} = await this.request({
action: 'unpublishedEntryDataFile',
params: {
branch: this.branch,
collection,
slug,
path,
id
}
});
return data;
}
async unpublishedEntryMediaFile(collection, slug, path, id) {
const file = await this.request({
action: 'unpublishedEntryMediaFile',
params: {
branch: this.branch,
collection,
slug,
path,
id
}
});
return deserializeMediaFile(file);
}
deleteUnpublishedEntry(collection, slug) {
return this.request({
action: 'deleteUnpublishedEntry',
params: {
branch: this.branch,
collection,
slug
}
});
}
async persistEntry(entry, options) {
const assets = await Promise.all(entry.assets.map(serializeAsset));
return this.request({
action: 'persistEntry',
params: {
branch: this.branch,
dataFiles: entry.dataFiles,
assets,
options: {
...options,
status: options.status || this.options.initialWorkflowStatus
},
cmsLabelPrefix: this.cmsLabelPrefix
}
});
}
updateUnpublishedEntryStatus(collection, slug, newStatus) {
return this.request({
action: 'updateUnpublishedEntryStatus',
params: {
branch: this.branch,
collection,
slug,
newStatus,
cmsLabelPrefix: this.cmsLabelPrefix
}
});
}
publishUnpublishedEntry(collection, slug) {
return this.request({
action: 'publishUnpublishedEntry',
params: {
branch: this.branch,
collection,
slug
}
});
}
async getMedia(mediaFolder = this.mediaFolder) {
const files = await this.request({
action: 'getMedia',
params: {
branch: this.branch,
mediaFolder
}
});
return files.map(deserializeMediaFile);
}
async getMediaFile(path) {
const file = await this.request({
action: 'getMediaFile',
params: {
branch: this.branch,
path
}
});
return deserializeMediaFile(file);
}
async persistMedia(assetProxy, options) {
const asset = await serializeAsset(assetProxy);
const file = await this.request({
action: 'persistMedia',
params: {
branch: this.branch,
asset,
options: {
commitMessage: options.commitMessage
}
}
});
return deserializeMediaFile(file);
}
deleteFiles(paths, commitMessage) {
return this.request({
action: 'deleteFiles',
params: {
branch: this.branch,
paths,
options: {
commitMessage
}
}
});
}
getDeployPreview(collection, slug) {
return this.request({
action: 'getDeployPreview',
params: {
branch: this.branch,
collection,
slug
}
});
}
}