2016-12-23 16:59:48 -02:00
|
|
|
import semaphore from "semaphore";
|
|
|
|
import AuthenticationPage from "./AuthenticationPage";
|
|
|
|
import API from "./API";
|
2017-04-14 19:19:45 +01:00
|
|
|
import { fileExtension } from '../../lib/pathHelper'
|
2016-06-05 01:52:18 -07:00
|
|
|
|
2016-09-04 19:55:14 +02:00
|
|
|
const MAX_CONCURRENT_DOWNLOADS = 10;
|
|
|
|
|
2016-06-05 01:52:18 -07:00
|
|
|
export default class GitHub {
|
2016-12-23 16:59:48 -02:00
|
|
|
constructor(config, proxied = false) {
|
2016-06-05 01:52:18 -07:00
|
|
|
this.config = config;
|
2016-12-23 16:59:48 -02:00
|
|
|
|
|
|
|
if (!proxied && config.getIn(["backend", "repo"]) == null) {
|
|
|
|
throw new Error("The GitHub backend needs a \"repo\" in the backend configuration.");
|
2016-06-05 01:52:18 -07:00
|
|
|
}
|
2016-12-23 16:59:48 -02:00
|
|
|
|
|
|
|
this.repo = config.getIn(["backend", "repo"], "");
|
|
|
|
this.branch = config.getIn(["backend", "branch"], "master");
|
2017-01-10 22:23:22 -02:00
|
|
|
this.token = '';
|
2016-06-05 01:52:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
authComponent() {
|
|
|
|
return AuthenticationPage;
|
|
|
|
}
|
|
|
|
|
|
|
|
setUser(user) {
|
2017-01-10 22:23:22 -02:00
|
|
|
this.token = user.token;
|
|
|
|
this.api = new API({ token: this.token, branch: this.branch, repo: this.repo });
|
2016-06-05 01:52:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
authenticate(state) {
|
2017-01-10 22:23:22 -02:00
|
|
|
this.token = state.token;
|
|
|
|
this.api = new API({ token: this.token, branch: this.branch, repo: this.repo });
|
2016-06-05 01:52:18 -07:00
|
|
|
return this.api.user().then((user) => {
|
|
|
|
user.token = state.token;
|
|
|
|
return user;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-01-10 22:23:22 -02:00
|
|
|
getToken() {
|
|
|
|
return Promise.resolve(this.token);
|
|
|
|
}
|
|
|
|
|
2017-04-14 19:19:45 +01:00
|
|
|
entriesByFolder(collection, extension) {
|
2016-12-23 16:59:48 -02:00
|
|
|
return this.api.listFiles(collection.get("folder"))
|
2017-04-19 11:44:04 -04:00
|
|
|
.then(files => files.filter(file => fileExtension(file.name) === extension))
|
2016-10-27 13:12:18 -02:00
|
|
|
.then(this.fetchFiles);
|
2016-06-05 01:52:18 -07:00
|
|
|
}
|
|
|
|
|
2016-10-27 14:23:36 +02:00
|
|
|
entriesByFiles(collection) {
|
2016-12-23 16:59:48 -02:00
|
|
|
const files = collection.get("files").map(collectionFile => ({
|
|
|
|
path: collectionFile.get("file"),
|
|
|
|
label: collectionFile.get("label"),
|
2016-10-27 14:23:36 +02:00
|
|
|
}));
|
2016-10-27 13:12:18 -02:00
|
|
|
return this.fetchFiles(files);
|
|
|
|
}
|
|
|
|
|
|
|
|
fetchFiles = (files) => {
|
2016-10-21 20:42:14 -02:00
|
|
|
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 });
|
2016-10-21 20:42:14 -02:00
|
|
|
sem.leave();
|
|
|
|
}).catch((err) => {
|
|
|
|
sem.leave();
|
|
|
|
reject(err);
|
2016-10-27 13:12:18 -02:00
|
|
|
}))
|
|
|
|
)));
|
2016-10-21 20:42:14 -02:00
|
|
|
});
|
|
|
|
return Promise.all(promises);
|
2016-10-27 13:12:18 -02:00
|
|
|
};
|
2016-07-19 17:11:22 -03:00
|
|
|
|
2016-10-10 15:34:21 -03:00
|
|
|
// Fetches a single entry.
|
|
|
|
getEntry(collection, slug, path) {
|
2016-10-27 14:23:36 +02:00
|
|
|
return this.api.readFile(path).then(data => ({
|
|
|
|
file: { path },
|
|
|
|
data,
|
|
|
|
}));
|
2016-10-10 15:34:21 -03:00
|
|
|
}
|
|
|
|
|
2016-08-29 17:09:04 -03:00
|
|
|
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) => {
|
2016-12-23 16:59:48 -02:00
|
|
|
const slug = branch.ref.split("refs/heads/cms/").pop();
|
2016-10-27 15:27:39 -02:00
|
|
|
return sem.take(() => this.api.readUnpublishedBranchFile(slug).then((data) => {
|
2016-10-10 18:33:49 -03:00
|
|
|
if (data === null || data === undefined) {
|
|
|
|
resolve(null);
|
|
|
|
sem.leave();
|
|
|
|
} else {
|
2017-01-11 20:58:15 -02:00
|
|
|
const path = data.metaData.objects.entry.path;
|
2016-10-27 15:27:39 -02:00
|
|
|
resolve({
|
|
|
|
slug,
|
|
|
|
file: { path },
|
|
|
|
data: data.fileData,
|
|
|
|
metaData: data.metaData,
|
2017-03-15 18:47:18 -07:00
|
|
|
isModification: data.isModification,
|
2016-10-27 15:27:39 -02:00
|
|
|
});
|
2016-10-10 18:33:49 -03:00
|
|
|
sem.leave();
|
|
|
|
}
|
2016-09-06 17:18:27 -03:00
|
|
|
}).catch((err) => {
|
|
|
|
sem.leave();
|
2017-01-11 20:58:15 -02:00
|
|
|
resolve(null);
|
2016-09-06 17:18:27 -03:00
|
|
|
}));
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
return Promise.all(promises);
|
2016-12-01 19:59:29 -02:00
|
|
|
})
|
|
|
|
.catch((error) => {
|
2016-12-23 16:59:48 -02:00
|
|
|
if (error.message === "Not Found") {
|
2016-12-01 19:59:29 -02:00
|
|
|
return Promise.resolve([]);
|
|
|
|
}
|
|
|
|
return error;
|
2016-10-28 11:42:31 -02:00
|
|
|
});
|
2016-09-06 13:04:17 -03:00
|
|
|
}
|
2016-09-13 03:59:48 -03:00
|
|
|
|
|
|
|
unpublishedEntry(collection, slug) {
|
2016-10-28 11:42:31 -02:00
|
|
|
return this.api.readUnpublishedBranchFile(slug)
|
2017-01-11 20:58:15 -02:00
|
|
|
.then((data) => {
|
|
|
|
if (!data) return null;
|
|
|
|
return {
|
|
|
|
slug,
|
|
|
|
file: { path: data.metaData.objects.entry.path },
|
|
|
|
data: data.fileData,
|
|
|
|
metaData: data.metaData,
|
2017-03-15 18:47:18 -07:00
|
|
|
isModification: data.isModification,
|
2017-01-11 20:58:15 -02:00
|
|
|
};
|
|
|
|
});
|
2016-09-13 03:59:48 -03:00
|
|
|
}
|
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
|
|
|
|
2017-03-11 13:47:36 -05:00
|
|
|
deleteUnpublishedEntry(collection, slug) {
|
|
|
|
return this.api.deleteUnpublishedEntry(collection, slug);
|
|
|
|
}
|
2017-01-11 20:58:15 -02:00
|
|
|
publishUnpublishedEntry(collection, slug) {
|
|
|
|
return this.api.publishUnpublishedEntry(collection, slug);
|
2016-09-14 18:25:45 -03:00
|
|
|
}
|
2016-06-05 01:52:18 -07:00
|
|
|
}
|