import arrayContainsObject from './arrayContainsObject';
import decode from './decode';
import encode from './encode';
import CoreManager from './CoreManager';
import type ParseObject from './ParseObject';
import type Pointer from './ParseObject';
import ParseRelation from './ParseRelation';
import unique from './unique';
export function opFromJSON(json: Record<string, any>): Op | null {
if (!json || !json.__op) {
return null;
}
switch (json.__op) {
case 'Delete':
return new UnsetOp();
case 'Increment':
return new IncrementOp(json.amount);
case 'Add':
return new AddOp(decode(json.objects));
case 'AddUnique':
return new AddUniqueOp(decode(json.objects));
case 'Remove':
return new RemoveOp(decode(json.objects));
case 'AddRelation': {
const toAdd = decode(json.objects);
if (!Array.isArray(toAdd)) {
return new RelationOp([], []);
}
return new RelationOp(toAdd, []);
}
case 'RemoveRelation': {
const toRemove = decode(json.objects);
if (!Array.isArray(toRemove)) {
return new RelationOp([], []);
}
return new RelationOp([], toRemove);
}
case 'Batch': {
let toAdd = [];
let toRemove = [];
for (let i = 0; i < json.ops.length; i++) {
if (json.ops[i].__op === 'AddRelation') {
toAdd = toAdd.concat(decode(json.ops[i].objects));
} else if (json.ops[i].__op === 'RemoveRelation') {
toRemove = toRemove.concat(decode(json.ops[i].objects));
}
}
return new RelationOp(toAdd, toRemove);
}
}
return null;
}
export class Op {
applyTo(value: any): any {}
mergeWith(previous: Op): Op | void {}
toJSON(offline?: boolean): any {}
}
export class SetOp extends Op {
_value: any;
constructor(value: any) {
super();
this._value = value;
}
applyTo(): any {
return this._value;
}
mergeWith(): SetOp {
return new SetOp(this._value);
}
toJSON(offline?: boolean): any {
return encode(this._value, false, true, undefined, offline);
}
}
export class UnsetOp extends Op {
applyTo() {
return undefined;
}
mergeWith(): UnsetOp {
return new UnsetOp();
}
toJSON(): { __op: string } {
return { __op: 'Delete' };
}
}
export class IncrementOp extends Op {
_amount: number;
constructor(amount: number) {
super();
if (typeof amount !== 'number') {
throw new TypeError('Increment Op must be initialized with a numeric amount.');
}
this._amount = amount;
}
applyTo(value: any): number {
if (typeof value === 'undefined') {
return this._amount;
}
if (typeof value !== 'number') {
throw new TypeError('Cannot increment a non-numeric value.');
}
return this._amount + value;
}
mergeWith(previous: Op): Op {
if (!previous) {
return this;
}
if (previous instanceof SetOp) {
return new SetOp(this.applyTo(previous._value));
}
if (previous instanceof UnsetOp) {
return new SetOp(this._amount);
}
if (previous instanceof IncrementOp) {
return new IncrementOp(this.applyTo(previous._amount));
}
throw new Error('Cannot merge Increment Op with the previous Op');
}
toJSON(): { __op: string; amount: number } {
return { __op: 'Increment', amount: this._amount };
}
}
export class AddOp extends Op {
_value: any[];
constructor(value: any | any[]) {
super();
this._value = Array.isArray(value) ? value : [value];
}
applyTo(value: any): any[] {
if (value == null) {
return this._value;
}
if (Array.isArray(value)) {
return value.concat(this._value);
}
throw new Error('Cannot add elements to a non-array value');
}
mergeWith(previous: Op): Op {
if (!previous) {
return this;
}
if (previous instanceof SetOp) {
return new SetOp(this.applyTo(previous._value));
}
if (previous instanceof UnsetOp) {
return new SetOp(this._value);
}
if (previous instanceof AddOp) {
return new AddOp(this.applyTo(previous._value));
}
throw new Error('Cannot merge Add Op with the previous Op');
}
toJSON(): { __op: string; objects: any } {
return { __op: 'Add', objects: encode(this._value, false, true) };
}
}
export class AddUniqueOp extends Op {
_value: any[];
constructor(value: any | any[]) {
super();
this._value = unique(Array.isArray(value) ? value : [value]);
}
applyTo(value: any | any[]): any[] {
if (value == null) {
return this._value || [];
}
if (Array.isArray(value)) {
const ParseObject = CoreManager.getParseObject();
const toAdd: any[] = [];
this._value.forEach(v => {
if (v instanceof ParseObject) {
if (!arrayContainsObject(value, v)) {
toAdd.push(v);
}
} else {
if (value.indexOf(v) < 0) {
toAdd.push(v);
}
}
});
return value.concat(toAdd);
}
throw new Error('Cannot add elements to a non-array value');
}
mergeWith(previous: Op): Op {
if (!previous) {
return this;
}
if (previous instanceof SetOp) {
return new SetOp(this.applyTo(previous._value));
}
if (previous instanceof UnsetOp) {
return new SetOp(this._value);
}
if (previous instanceof AddUniqueOp) {
return new AddUniqueOp(this.applyTo(previous._value));
}
throw new Error('Cannot merge AddUnique Op with the previous Op');
}
toJSON(): { __op: string; objects: any } {
return { __op: 'AddUnique', objects: encode(this._value, false, true) };
}
}
export class RemoveOp extends Op {
_value: any[];
constructor(value: any | any[]) {
super();
this._value = unique(Array.isArray(value) ? value : [value]);
}
applyTo(value: any | any[]): any[] {
if (value == null) {
return [];
}
if (Array.isArray(value)) {
const ParseObject = CoreManager.getParseObject();
const removed = value.concat([]);
for (let i = 0; i < this._value.length; i++) {
let index = removed.indexOf(this._value[i]);
while (index > -1) {
removed.splice(index, 1);
index = removed.indexOf(this._value[i]);
}
if (this._value[i] instanceof ParseObject && this._value[i].id) {
for (let j = 0; j < removed.length; j++) {
if (removed[j] instanceof ParseObject && this._value[i].id === removed[j].id) {
removed.splice(j, 1);
j--;
}
}
}
}
return removed;
}
throw new Error('Cannot remove elements from a non-array value');
}
mergeWith(previous: Op): Op {
if (!previous) {
return this;
}
if (previous instanceof SetOp) {
return new SetOp(this.applyTo(previous._value));
}
if (previous instanceof UnsetOp) {
return new UnsetOp();
}
if (previous instanceof RemoveOp) {
const ParseObject = CoreManager.getParseObject();
const uniques = previous._value.concat([]);
for (let i = 0; i < this._value.length; i++) {
if (this._value[i] instanceof ParseObject) {
if (!arrayContainsObject(uniques, this._value[i])) {
uniques.push(this._value[i]);
}
} else {
if (uniques.indexOf(this._value[i]) < 0) {
uniques.push(this._value[i]);
}
}
}
return new RemoveOp(uniques);
}
throw new Error('Cannot merge Remove Op with the previous Op');
}
toJSON(): { __op: string; objects: any } {
return { __op: 'Remove', objects: encode(this._value, false, true) };
}
}
export class RelationOp extends Op {
_targetClassName: string | null;
relationsToAdd: string[];
relationsToRemove: string[];
constructor(adds: (ParseObject | string)[], removes: (ParseObject | string)[]) {
super();
this._targetClassName = null;
if (Array.isArray(adds)) {
this.relationsToAdd = unique(adds.map(this._extractId, this));
}
if (Array.isArray(removes)) {
this.relationsToRemove = unique(removes.map(this._extractId, this));
}
}
_extractId(obj: string | ParseObject): string {
if (typeof obj === 'string') {
return obj;
}
if (!obj.id) {
throw new Error('You cannot add or remove an unsaved Parse Object from a relation');
}
if (!this._targetClassName) {
this._targetClassName = obj.className;
}
if (this._targetClassName !== obj.className) {
throw new Error(
'Tried to create a Relation with 2 different object types: ' +
this._targetClassName +
' and ' +
obj.className +
'.'
);
}
return obj.id;
}
applyTo(value: any, parent?: ParseObject, key?: string): ParseRelation {
if (!value) {
if (!parent || !key) {
throw new Error(
'Cannot apply a RelationOp without either a previous value, or an object and a key'
);
}
const relation = new ParseRelation(parent, key);
relation.targetClassName = this._targetClassName;
return relation;
}
if (value instanceof ParseRelation) {
if (this._targetClassName) {
if (value.targetClassName) {
if (this._targetClassName !== value.targetClassName) {
throw new Error(
'Related object must be a ' +
value.targetClassName +
', but a ' +
this._targetClassName +
' was passed in.'
);
}
} else {
value.targetClassName = this._targetClassName;
}
}
return value;
} else {
throw new Error('Relation cannot be applied to a non-relation field');
}
}
mergeWith(previous: Op): Op {
if (!previous) {
return this;
} else if (previous instanceof UnsetOp) {
throw new Error('You cannot modify a relation after deleting it.');
} else if (previous instanceof SetOp && previous._value instanceof ParseRelation) {
return this;
} else if (previous instanceof RelationOp) {
if (previous._targetClassName && previous._targetClassName !== this._targetClassName) {
throw new Error(
'Related object must be of class ' +
previous._targetClassName +
', but ' +
(this._targetClassName || 'null') +
' was passed in.'
);
}
const newAdd = previous.relationsToAdd.concat([]);
this.relationsToRemove.forEach(r => {
const index = newAdd.indexOf(r);
if (index > -1) {
newAdd.splice(index, 1);
}
});
this.relationsToAdd.forEach(r => {
const index = newAdd.indexOf(r);
if (index < 0) {
newAdd.push(r);
}
});
const newRemove = previous.relationsToRemove.concat([]);
this.relationsToAdd.forEach(r => {
const index = newRemove.indexOf(r);
if (index > -1) {
newRemove.splice(index, 1);
}
});
this.relationsToRemove.forEach(r => {
const index = newRemove.indexOf(r);
if (index < 0) {
newRemove.push(r);
}
});
const newRelation = new RelationOp(newAdd, newRemove);
newRelation._targetClassName = this._targetClassName;
return newRelation;
}
throw new Error('Cannot merge Relation Op with the previous Op');
}
toJSON(): { __op?: string; objects?: any; ops?: any } {
const idToPointer = (id: string) => {
return {
__type: 'Pointer',
className: this._targetClassName,
objectId: id,
};
};
let pointers: any = null;
let adds: null | { __op: string; objects: null | Pointer[] } = null;
let removes: null | { __op: string; objects: null | Pointer[] } = null;
if (this.relationsToAdd.length > 0) {
pointers = this.relationsToAdd.map(idToPointer);
adds = { __op: 'AddRelation', objects: pointers };
}
if (this.relationsToRemove.length > 0) {
pointers = this.relationsToRemove.map(idToPointer);
removes = { __op: 'RemoveRelation', objects: pointers };
}
if (adds && removes) {
return { __op: 'Batch', ops: [adds, removes] };
}
return adds || removes || {};
}
}
CoreManager.setParseOp({
Op,
opFromJSON,
SetOp,
UnsetOp,
IncrementOp,
AddOp,
RelationOp,
RemoveOp,
AddUniqueOp,
});