2018-12-17 08:45:09 -05:00
|
|
|
const Promise = require('bluebird');
|
2019-12-16 03:29:24 -05:00
|
|
|
const dbBackup = require('../../data/db/backup');
|
2018-12-17 08:45:09 -05:00
|
|
|
const exporter = require('../../data/exporter');
|
|
|
|
const importer = require('../../data/importer');
|
2020-05-22 13:22:20 -05:00
|
|
|
const errors = require('@tryghost/errors');
|
2018-12-17 08:45:09 -05:00
|
|
|
const models = require('../../models');
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
docName: 'db',
|
|
|
|
|
|
|
|
backupContent: {
|
|
|
|
permissions: true,
|
|
|
|
options: [
|
|
|
|
'include',
|
|
|
|
'filename'
|
|
|
|
],
|
|
|
|
validation: {
|
|
|
|
options: {
|
|
|
|
include: {
|
|
|
|
values: exporter.EXCLUDED_TABLES
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
query(frame) {
|
|
|
|
// NOTE: we need to have `include` property available as backupDatabase uses it internally
|
|
|
|
Object.assign(frame.options, {include: frame.options.withRelated});
|
|
|
|
|
2019-12-16 03:29:24 -05:00
|
|
|
return dbBackup.backup(frame.options);
|
2018-12-17 08:45:09 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
exportContent: {
|
|
|
|
options: [
|
|
|
|
'include'
|
|
|
|
],
|
|
|
|
validation: {
|
|
|
|
options: {
|
|
|
|
include: {
|
|
|
|
values: exporter.EXCLUDED_TABLES
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
headers: {
|
|
|
|
disposition: {
|
|
|
|
type: 'file',
|
|
|
|
value: () => (exporter.fileName())
|
|
|
|
}
|
|
|
|
},
|
2019-08-02 04:28:02 -05:00
|
|
|
permissions: true,
|
2018-12-17 08:45:09 -05:00
|
|
|
query(frame) {
|
|
|
|
return Promise.resolve()
|
|
|
|
.then(() => exporter.doExport({include: frame.options.withRelated}))
|
|
|
|
.catch((err) => {
|
2020-05-22 13:22:20 -05:00
|
|
|
return Promise.reject(new errors.GhostError({err: err}));
|
2018-12-17 08:45:09 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
importContent: {
|
|
|
|
options: [
|
|
|
|
'include'
|
|
|
|
],
|
|
|
|
validation: {
|
|
|
|
options: {
|
|
|
|
include: {
|
|
|
|
values: exporter.EXCLUDED_TABLES
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2019-08-02 04:28:02 -05:00
|
|
|
permissions: true,
|
2018-12-17 08:45:09 -05:00
|
|
|
query(frame) {
|
2018-12-20 05:01:48 -05:00
|
|
|
return importer.importFromFile(frame.file, {include: frame.options.withRelated});
|
2018-12-17 08:45:09 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
deleteAllContent: {
|
|
|
|
statusCode: 204,
|
2019-08-02 04:28:02 -05:00
|
|
|
permissions: true,
|
2018-12-17 08:45:09 -05:00
|
|
|
query() {
|
|
|
|
/**
|
|
|
|
* @NOTE:
|
|
|
|
* We fetch all posts with `columns:id` to increase the speed of this endpoint.
|
|
|
|
* And if you trigger `post.destroy(..)`, this will trigger bookshelf and model events.
|
|
|
|
* But we only have to `id` available in the model. This won't work, because:
|
|
|
|
* - model layer can't trigger event e.g. `post.page` to trigger `post|page.unpublished`.
|
|
|
|
* - `onDestroyed` or `onDestroying` can contain custom logic
|
|
|
|
*/
|
|
|
|
function deleteContent() {
|
|
|
|
return models.Base.transaction((transacting) => {
|
|
|
|
const queryOpts = {
|
|
|
|
columns: 'id',
|
|
|
|
context: {internal: true},
|
|
|
|
destroyAll: true,
|
|
|
|
transacting: transacting
|
|
|
|
};
|
|
|
|
|
|
|
|
return models.Post.findAll(queryOpts)
|
|
|
|
.then((response) => {
|
|
|
|
return Promise.map(response.models, (post) => {
|
|
|
|
return models.Post.destroy(Object.assign({id: post.id}, queryOpts));
|
|
|
|
}, {concurrency: 100});
|
|
|
|
})
|
|
|
|
.then(() => models.Tag.findAll(queryOpts))
|
|
|
|
.then((response) => {
|
|
|
|
return Promise.map(response.models, (tag) => {
|
|
|
|
return models.Tag.destroy(Object.assign({id: tag.id}, queryOpts));
|
|
|
|
}, {concurrency: 100});
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
2020-05-22 13:22:20 -05:00
|
|
|
throw new errors.GhostError({
|
2018-12-17 08:45:09 -05:00
|
|
|
err: err
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-12-16 03:29:24 -05:00
|
|
|
return dbBackup.backup().then(deleteContent);
|
2018-12-17 08:45:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|