static-cms/src/backends/github/implementation.js

123 lines
3.4 KiB
JavaScript
Raw Normal View History

2016-09-04 19:55:14 +02:00
import semaphore from 'semaphore';
2016-09-06 13:04:17 -03:00
import { createEntry } from '../../valueObjects/Entry';
import AuthenticationPage from './AuthenticationPage';
import API from './API';
2016-09-04 19:55:14 +02:00
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) {
2016-10-27 13:12:18 -02:00
return this.api.listFiles(collection.get('folder'))
.then(this.fetchFiles);
}
entriesByFiles(collection) {
const files = collection.get('files').map(collectionFile => ({
path: collectionFile.get('file'),
label: collectionFile.get('label'),
}));
2016-10-27 13:12:18 -02:00
return this.fetchFiles(files);
}
fetchFiles = (files) => {
const sem = semaphore(MAX_CONCURRENT_DOWNLOADS);
const promises = [];
files.forEach((file) => {
2016-10-27 13:12:18 -02:00
promises.push(new Promise((resolve, reject) => (
sem.take(() => this.api.readFile(file.path, file.sha).then((data) => {
resolve({ file, data });
sem.leave();
}).catch((err) => {
sem.leave();
reject(err);
2016-10-27 13:12:18 -02:00
}))
)));
});
return Promise.all(promises);
2016-10-27 13:12:18 -02:00
};
2016-07-19 17:11:22 -03:00
// 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);
2016-07-19 17:11:22 -03:00
}
2016-09-06 13:04:17 -03:00
unpublishedEntries() {
2016-09-06 17:18:27 -03:00
return this.api.listUnpublishedBranches().then((branches) => {
const sem = semaphore(MAX_CONCURRENT_DOWNLOADS);
const promises = [];
branches.map((branch) => {
promises.push(new Promise((resolve, reject) => {
const slug = branch.ref.split('refs/heads/cms/').pop();
return sem.take(() => this.api.readUnpublishedBranchFile(slug).then((data) => {
if (data === null || data === undefined) {
resolve(null);
sem.leave();
} else {
const path = data.metaData.objects.entry;
resolve({
slug,
file: { path },
data: data.fileData,
metaData: data.metaData,
});
sem.leave();
}
2016-09-06 17:18:27 -03:00
}).catch((err) => {
sem.leave();
reject(err);
}));
}));
});
return Promise.all(promises);
})
2016-09-06 13:04:17 -03:00
}
unpublishedEntry(collection, slug) {
2016-10-10 16:10:55 -03:00
return this.unpublishedEntries().then(response => (
response.filter((entry) => {
return entry.metaData && entry.slug === slug;
})[0]
));
}
2016-09-13 16:00:24 -03:00
updateUnpublishedEntryStatus(collection, slug, newStatus) {
return this.api.updateUnpublishedEntryStatus(collection, slug, newStatus);
}
2016-09-14 18:25:45 -03:00
publishUnpublishedEntry(collection, slug, status) {
return this.api.publishUnpublishedEntry(collection, slug, status);
}
}