Drop _Join collection when deleting a relation field

This commit is contained in:
Drew Gross
2016-02-09 20:16:53 -08:00
parent f07836e33f
commit b0c4b8f6ce
3 changed files with 103 additions and 4 deletions

View File

@@ -43,8 +43,6 @@ ExportAdapter.prototype.connect = function() {
// Returns a promise for a Mongo collection.
// Generally just for internal use.
var joinRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;
var otherRegex = /^[A-Za-z][A-Za-z0-9_]*$/;
ExportAdapter.prototype.collection = function(className) {
if (!Schema.classNameIsValid(className)) {
throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME,

View File

@@ -413,7 +413,11 @@ Schema.prototype.validateField = function(className, key, type, freeze) {
// to remove unused fields, if other writers are writing objects that include
// this field, the field may reappear. Returns a Promise that resolves with
// no object on success, or rejects with { code, error } on failure.
Schema.prototype.deleteField = function(fieldName, className) {
// Passing the database and prefix is necessary in order to drop relation collections
// and remove fields from objects. Ideally the database would belong to
// a database adapter and this fuction would close over it or access it via member.
Schema.prototype.deleteField = function(fieldName, className, database, prefix) {
if (!classNameIsValid(className)) {
return Promise.reject({
code: Parse.Error.INVALID_CLASS_NAME,
@@ -438,6 +442,37 @@ Schema.prototype.deleteField = function(fieldName, className) {
return this.reload()
.then(schema => {
return schema.hasClass(className)
.then(hasClass => {
if (!hasClass) {
return Promise.reject({
code: Parse.Error.INVALID_CLASS_NAME,
error: 'class ' + className + ' does not exist',
});
}
if (!schema.data[className][fieldName]) {
return Promise.reject({
code: 255,
error: 'field ' + fieldName + ' does not exist, cannot delete',
});
}
let p = null;
if (schema.data[className][fieldName].startsWith('relation')) {
//For relations, drop the _Join table
p = database.dropCollection(prefix + '_Join:' + fieldName + ':' + className);
} else {
//for non-relations, remove all the data. This is necessary to ensure that the data is still gone
//if they add the same field.
p = Promise.resolve();
}
return p.then(() => {
//Save the _SCHEMA object
return Promise.resolve();
});
});
});
}
@@ -509,6 +544,13 @@ Schema.prototype.getExpectedType = function(className, key) {
return undefined;
};
// Checks if a given class is in the schema. Needs to load the
// schema first, which is kinda janky. Hopefully we can refactor
// and make this be a regular value. Parse would probably
Schema.prototype.hasClass = function(className) {
return this.reload().then(newSchema => !!newSchema.data[className]);
}
// Helper function to check if a field is a pointer, returns true or false.
Schema.prototype.isPointer = function(className, key) {
var expected = this.getExpectedType(className, key);