126 lines
3.6 KiB
JavaScript
126 lines
3.6 KiB
JavaScript
import semaphore from 'semaphore';
|
|
import { createEntry } from '../../valueObjects/Entry';
|
|
import AuthenticationPage from './AuthenticationPage';
|
|
import API from './API';
|
|
|
|
const MAX_CONCURRENT_DOWNLOADS = 10;
|
|
|
|
export default class GitHub {
|
|
constructor(config) {
|
|
this.config = config;
|
|
if (config.getIn(['backend', 'repo']) == null) {
|
|
throw 'The GitHub backend needs a "repo" in the backend configuration.';
|
|
}
|
|
this.repo = config.getIn(['backend', 'repo']);
|
|
this.branch = config.getIn(['backend', 'branch']) || 'master';
|
|
}
|
|
|
|
authComponent() {
|
|
return AuthenticationPage;
|
|
}
|
|
|
|
setUser(user) {
|
|
this.api = new API(user.token, this.repo, this.branch || 'master');
|
|
}
|
|
|
|
authenticate(state) {
|
|
this.api = new API(state.token, this.repo, this.branch || 'master');
|
|
return this.api.user().then((user) => {
|
|
user.token = state.token;
|
|
return user;
|
|
});
|
|
}
|
|
|
|
entriesByFolder(collection) {
|
|
return this.api.listFiles(collection.get('folder')).then(files => this.entriesByFiles(collection, files));
|
|
}
|
|
|
|
entriesByFiles(collection) {
|
|
const files = collection.get('files').map(collectionFile => ({
|
|
path: collectionFile.get('file'),
|
|
label: collectionFile.get('label'),
|
|
}));
|
|
const sem = semaphore(MAX_CONCURRENT_DOWNLOADS);
|
|
const promises = [];
|
|
files.forEach((file) => {
|
|
promises.push(new Promise((resolve, reject) => {
|
|
return sem.take(() => this.api.readFile(file.path, file.sha).then((data) => {
|
|
resolve(
|
|
{
|
|
file,
|
|
data,
|
|
}
|
|
);
|
|
sem.leave();
|
|
}).catch((err) => {
|
|
sem.leave();
|
|
reject(err);
|
|
}));
|
|
}));
|
|
});
|
|
return Promise.all(promises);
|
|
}
|
|
|
|
// Fetches a single entry.
|
|
getEntry(collection, slug, path) {
|
|
return this.api.readFile(path).then(data => ({
|
|
file: { path },
|
|
data,
|
|
}));
|
|
}
|
|
|
|
persistEntry(entry, mediaFiles = [], options = {}) {
|
|
return this.api.persistFiles(entry, mediaFiles, options);
|
|
}
|
|
|
|
unpublishedEntries() {
|
|
return this.api.listUnpublishedBranches().then((branches) => {
|
|
const sem = semaphore(MAX_CONCURRENT_DOWNLOADS);
|
|
const promises = [];
|
|
branches.map((branch) => {
|
|
promises.push(new Promise((resolve, reject) => {
|
|
const contentKey = branch.ref.split('refs/heads/cms/').pop();
|
|
return sem.take(() => this.api.readUnpublishedBranchFile(contentKey).then((data) => {
|
|
if (data === null || data === undefined) {
|
|
resolve(null);
|
|
sem.leave();
|
|
} else {
|
|
const entryPath = data.metaData.objects.entry;
|
|
const entry = createEntry('draft', contentKey, entryPath, { raw: data.file });
|
|
entry.metaData = data.metaData;
|
|
resolve(entry);
|
|
sem.leave();
|
|
}
|
|
}).catch((err) => {
|
|
sem.leave();
|
|
reject(err);
|
|
}));
|
|
}));
|
|
});
|
|
return Promise.all(promises);
|
|
}).then((entries) => {
|
|
const filteredEntries = entries.filter(entry => entry !== null);
|
|
return {
|
|
pagination: 0,
|
|
entries: filteredEntries,
|
|
};
|
|
});
|
|
}
|
|
|
|
unpublishedEntry(collection, slug) {
|
|
return this.unpublishedEntries().then(response => (
|
|
response.entries.filter((entry) => {
|
|
return entry.metaData && entry.slug === slug;
|
|
})[0]
|
|
));
|
|
}
|
|
|
|
updateUnpublishedEntryStatus(collection, slug, newStatus) {
|
|
return this.api.updateUnpublishedEntryStatus(collection, slug, newStatus);
|
|
}
|
|
|
|
publishUnpublishedEntry(collection, slug, status) {
|
|
return this.api.publishUnpublishedEntry(collection, slug, status);
|
|
}
|
|
}
|