OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/frontend/app/components/api/api-v3/hal-resources/work-package-resource.servi...

666 lines
20 KiB

//-- copyright
// OpenProject is a project management system.
// Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License version 3.
//
// OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
// Copyright (C) 2006-2013 Jean-Philippe Lang
// Copyright (C) 2010-2013 the ChiliProject Team
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
//
// See doc/COPYRIGHT.rdoc for more details.
//++
import {HalResource} from './hal-resource.service';
import {opApiModule} from '../../../../angular-modules';
import {WorkPackageCacheService} from '../../../work-packages/work-package-cache.service';
import {SchemaCacheService} from './../../../schemas/schema-cache.service';
import {ApiWorkPackagesService} from '../../api-work-packages/api-work-packages.service';
import {CollectionResource, CollectionResourceInterface} from './collection-resource.service';
import {AttachmentCollectionResourceInterface} from './attachment-collection-resource.service';
import {UploadFile} from '../../op-file-upload/op-file-upload.service';
import IQService = angular.IQService;
import IPromise = angular.IPromise;
import ITimeoutService = angular.ITimeoutService;
import {States} from '../../../states.service';
import {SchemaResource} from './schema-resource.service';
import {TypeResource} from './type-resource.service';
import {RelationResourceInterface} from './relation-resource.service';
import {WorkPackageCreateService} from '../../../wp-create/wp-create.service';
import {WorkPackageNotificationService} from '../../../wp-edit/wp-notification.service';
interface WorkPackageResourceEmbedded {
activities:CollectionResourceInterface;
ancestors:WorkPackageResourceInterface[];
assignee:HalResource | any;
attachments:AttachmentCollectionResourceInterface;
author:HalResource | any;
availableWatchers:HalResource | any;
category:HalResource | any;
children:WorkPackageResourceInterface[];
parent:HalResource | any;
priority:HalResource | any;
project:HalResource | any;
relations:CollectionResourceInterface;
responsible:HalResource | any;
revisions:CollectionResourceInterface | any;
status:HalResource | any;
timeEntries:HalResource[] | any[];
type:TypeResource;
version:HalResource | any;
watchers:CollectionResourceInterface;
// For regular work packages
startDate:string;
dueDate:string;
// Only for milestones
date:string;
relatedBy:RelationResourceInterface | null;
}
interface WorkPackageResourceLinks extends WorkPackageResourceEmbedded {
addAttachment(attachment:HalResource):ng.IPromise<any>;
addChild(child:HalResource):ng.IPromise<any>;
addComment(comment:HalResource):ng.IPromise<any>;
addRelation(relation:any):ng.IPromise<any>;
addWatcher(watcher:HalResource):ng.IPromise<any>;
changeParent(params:any):ng.IPromise<any>;
copy():ng.IPromise<WorkPackageResource>;
delete():ng.IPromise<any>;
logTime():ng.IPromise<any>;
move():ng.IPromise<any>;
removeWatcher():ng.IPromise<any>;
self():ng.IPromise<any>;
update(payload:any):ng.IPromise<any>;
updateImmediately(payload:any):ng.IPromise<any>;
watch():ng.IPromise<any>;
}
interface WorkPackageLinksObject extends WorkPackageResourceLinks {
schema:HalResource;
}
var $q:IQService;
var $stateParams:any;
var $timeout:ITimeoutService;
var I18n:op.I18n;
var states:States;
var apiWorkPackages:ApiWorkPackagesService;
var wpCacheService:WorkPackageCacheService;
var schemaCacheService:SchemaCacheService;
var NotificationsService:any;
var wpNotificationsService:WorkPackageNotificationService;
var wpCreate:WorkPackageCreateService;
var AttachmentCollectionResource:any;
var v3Path:any;
export class WorkPackageResource extends HalResource {
// Add index signature for getter this[attr]
[attribute:string]:any;
public $embedded:WorkPackageResourceEmbedded;
public $links:WorkPackageLinksObject;
public $pristine:{ [attribute:string]:any } = {};
public subject:string;
public updatedAt:Date;
public lockVersion:number;
public description:any;
public inFlight:boolean;
public activities:CollectionResourceInterface;
public attachments:AttachmentCollectionResourceInterface;
public pendingAttachments:UploadFile[] = [];
private form:any;
// Keep a reference to an embedded form schema,
// if this work package currently has one.
private overriddenSchema:SchemaResource | null;
public get id():string {
return this.$source.id || this.idFromLink;
}
public static idFromLink(href:string):string {
return href.split('/').pop()!;
}
public get idFromLink():string {
if (this.href) {
return WorkPackageResource.idFromLink(this.href);
}
return '';
}
/**
* Return the ids of all its ancestors, if any
*/
public get ancestorIds():string {
const ancestors = (this as any).ancestors;
return ancestors.map((el:WorkPackageResource) => el.id.toString());
}
/**
* Return "<type name>: <subject>" if the type is known.
*/
public get subjectWithType():string {
if (this.type) {
return `${this.type.name}: ${this.subject}`;
} else {
return this.subject;
}
}
public get isNew():boolean {
return this.id === 'new';
}
public get isMilestone():boolean {
return this.schema.hasOwnProperty('date');
}
/**
* Returns true if any field is in edition in this resource.
*/
public get dirty():boolean {
return this.modifiedFields.length > 0;
}
/**
* Returns all modified fields by comparing open $pristine fields.
*/
public get modifiedFields():string[] {
var modified:string[] = [];
angular.forEach(this.$pristine, (value, key) => {
var args = [this[key], value];
if ((this as any)[key] instanceof HalResource) {
args = args.map(arg => (arg ? arg.$source : arg));
}
if (!_.isEqual(args[0], args[1])) {
modified.push(key);
}
});
return modified;
}
public get isLeaf():boolean {
var children = this.$links.children;
return !(children && children.length > 0);
}
public get isEditable():boolean {
return !!this.$links.update || this.isNew;
}
/**
* Return whether the user is able to upload an attachment.
*
* If either the `addAttachment` link is provided or the resource is being created,
* adding attachments is allowed.
*/
public get canAddAttachments():boolean {
return !!this.$links.addAttachment || this.isNew;
}
/**
* Initialise the work package resource.
*
* Make the attachments an `AttachmentCollectionResource`. This should actually
* be done automatically, but the backend does not provide typed collections yet.
*/
protected $initialize(source:any) {
super.$initialize(source);
var attachments:{ $source:any, $loaded:boolean } = this.attachments || {
$source: void 0,
$loaded: false
};
this.attachments = new AttachmentCollectionResource(
attachments.$source,
attachments.$loaded
);
}
/**
* Remove the given attachment either from the pending attachments or from
* the attachment collection, if it is a resource.
*
* Removing it from the elements array assures that the view gets updated immediately.
* If an error occurs, the user gets notified and the attachment is pushed to the elements.
*/
public removeAttachment(attachment:any) {
if (attachment.$isHal) {
attachment.delete()
.then(() => {
this.updateAttachments();
})
.catch((error:any) => {
wpNotificationsService.handleErrorResponse(error, this as any);
this.attachments.elements.push(attachment);
});
}
_.pull(this.attachments.elements, attachment);
_.pull(this.pendingAttachments, attachment);
}
/**
* Upload the pending attachments if the work package exists.
* Do nothing, if the work package is being created.
*/
public uploadPendingAttachments():ng.IPromise<any> | void {
if (!this.pendingAttachments.length) {
return;
}
const attachments = this.pendingAttachments;
this.pendingAttachments = [];
return this.uploadAttachments(attachments);
}
/**
* Upload the given attachments, update the resource and notify the user.
* Return an updated AttachmentCollectionResource.
*/
public uploadAttachments(files:UploadFile[]):IPromise<any> {
const {uploads, finished} = this.attachments.upload(files);
const message = I18n.t('js.label_upload_notification', this);
const notification = NotificationsService.addWorkPackageUpload(message, uploads);
return finished
.then(() => {
$timeout(() => NotificationsService.remove(notification), 700);
return this.updateAttachments();
})
.catch(error => {
wpNotificationsService.handleRawError(error, this as any);
});
}
/**
* Uploads the attachments and reloads the work package when done
* Reloading is skipped if no attachment is added
*/
private uploadAttachmentsAndReload() {
const attachmentUpload = this.uploadPendingAttachments();
if (attachmentUpload) {
attachmentUpload.then((attachmentsResource) => {
if (attachmentsResource.count > 0) {
wpCacheService.loadWorkPackage(this.id, true);
}
});
}
}
public allowedValuesFor(field:string):ng.IPromise<HalResource[]> {
var deferred = $q.defer();
this.getForm().then((form:any) => {
const fieldSchema = form.$embedded.schema[field];
if (!fieldSchema) {
deferred.resolve([]);
} else if (fieldSchema.allowedValues && fieldSchema.allowedValues['$load']) {
let allowedValues = fieldSchema.allowedValues;
return allowedValues.$load().then((loadedValues:CollectionResource) => {
deferred.resolve(loadedValues.elements);
});
} else {
deferred.resolve(fieldSchema.allowedValues);
}
});
return deferred.promise;
}
public setAllowedValueFor(field:string, value:string | HalResource) {
return this.allowedValuesFor(field).then(allowedValues => {
let newValue;
if ((value as HalResource)['$href']) {
newValue = _.find(allowedValues,
(entry:any) => entry.$href === (value as HalResource).$href);
} else if (allowedValues) {
newValue = _.find(allowedValues, (entry:any) => entry === value);
} else {
newValue = value;
}
if (newValue) {
(this as any)[field] = newValue;
}
wpCacheService.updateWorkPackage(this as any);
});
}
public getForm() {
if (!this.form) {
this.updateForm(this.$source).catch(error => {
NotificationsService.addError(error.message);
});
}
return this.form;
}
public updateForm(payload:{[attribute:string]:any}) {
// Always resolve form to the latest form
// This way, we won't have to actively reset it.
// But store the existing form in case of an error.
// Because if we get an error, the object returned is not a form
// and thus lacks the links the implementation depends upon.
var oldForm = this.form;
this.form = this.$links.update(payload);
var deferred = $q.defer();
this.form
.then((form:any) => {
// Override the current schema with
// the changes from API
this.overriddenSchema = form.$embedded.schema;
// Take over new values from the form
// this resource doesn't know yet.
_.defaultsDeep(this.$source, form.$source._embedded.payload);
this.$initialize(this.$source);
deferred.resolve(form);
})
.catch((error:any) => {
this.form = oldForm;
deferred.reject(error);
});
return deferred.promise;
}
public loadFormSchema() {
return this.getForm().then((form:any) => {
this.overriddenSchema = form.$embedded.schema;
return this.schema;
});
}
public save():ng.IPromise<WorkPackageResourceInterface> {
var deferred = $q.defer();
this.inFlight = true;
const wasNew = this.isNew;
this.updateForm(this.$source)
.then(form => {
const payload = this.mergeWithForm(form);
const sentValues = Object.keys(this.$pristine);
this.$links.updateImmediately(payload)
.then((workPackage:WorkPackageResource) => {
// Remove the current form and schema, otherwise old form data
// might still be used for the next edit field to be edited
this.form = null;
this.overriddenSchema = null;
// Initialize any potentially new HAL values
this.$initialize(workPackage);
// Ensure the schema is loaded before updating
schemaCacheService.ensureLoaded(this).then(() => {
this.updateActivities();
if (wasNew) {
this.uploadAttachmentsAndReload();
wpCreate.newWorkPackageCreated(this as any);
}
// Remove only those pristine values that were submitted
angular.forEach(sentValues, (key) => {
delete this.$pristine[key];
});
wpCacheService.updateWorkPackage(this as any);
deferred.resolve(this);
});
})
.catch(error => {
deferred.reject(error);
wpCacheService.updateWorkPackage(this as any);
})
.finally(() => {
this.inFlight = false;
});
})
.catch(() => {
this.inFlight = false;
deferred.reject();
});
return deferred.promise;
}
public storePristine(attribute:string) {
if (this.$pristine.hasOwnProperty(attribute)) {
return;
}
this.$pristine[attribute] = angular.copy(this[attribute]);
}
public restoreFromPristine(attribute:string) {
if (this.$pristine.hasOwnProperty(attribute)) {
this[attribute] = this.$pristine[attribute];
delete this.$pristine[attribute];
}
}
public isParentOf(otherWorkPackage:WorkPackageResourceInterface) {
return otherWorkPackage.parent.$links.self.$link.href === this.$links.self.$link.href;
}
private mergeWithForm(form:any) {
var plainPayload = form.payload.$plain();
var schema = form.$embedded.schema;
// Merge embedded properties from form payload
// Do not use properties on this, since they may be incomplete
// e.g., when switching to a type that requires a custom field.
Object.keys(plainPayload).forEach(key => {
if (typeof(schema[key]) === 'object' && schema[key].writable === true) {
plainPayload[key] = this[key];
}
});
// Merged linked properties from form payload
Object.keys(plainPayload._links).forEach(key => {
if (typeof(schema[key]) === 'object' && schema[key].writable === true) {
var isArray = (schema[key].type || '').startsWith('[]');
if (isArray) {
var links:{ href:string }[] = [];
var val = this[key];
if (val) {
var elements = (val.forEach && val) || val.elements;
elements.forEach((link:{ href:string }) => {
if (link.href) {
links.push({href: link.href});
}
});
}
plainPayload._links[key] = links;
} else {
var value = this[key] ? this[key].href : null;
plainPayload._links[key] = {href: value};
}
}
});
return plainPayload;
}
/**
* Invalidate a set of linked resources of this work package.
* And inform the cache service about the work package update.
*
* Return a promise that returns the linked resources as properties.
* Return a rejected promise, if the resource is not a property of the work package.
*/
public updateLinkedResources(...resourceNames:string[]):ng.IPromise<any> {
const resources:{ [id:string]:IPromise<HalResource> } = {};
resourceNames.forEach(name => {
const linked = this[name];
resources[name] = linked ? linked.$update() : $q.reject();
});
const promise = $q.all(resources);
promise.then(() => {
wpCacheService.updateWorkPackage(this as any);
});
return promise;
}
/**
* Get updated activities from the server and inform the cache service about the work
* package update.
*
* Return a promise that returns the activities. Reject, if the work package has
* no activities.
*/
public updateActivities():IPromise<HalResource> {
return this
.updateLinkedResources('activities')
.then((resources:any) => resources.activities);
}
/**
* Get updated attachments and activities from the server and inform the cache service
* about the update.
*
* Return a promise that returns the attachments. Reject, if the work package has
* no attachments.
*/
public updateAttachments():IPromise<HalResource> {
return this
.updateLinkedResources('activities', 'attachments')
.then((resources:any) => resources.attachments);
}
/**
* Assign values from the form for a newly created work package resource.
* @param form
*/
public initializeNewResource(form:any) {
this.overriddenSchema = form.schema;
this.form = $q.when(form);
this.$source.id = 'new';
// Set update link to form
this['update'] = this.$links.update = form.$links.self;
// Use POST /work_packages for saving link
this['updateImmediately'] = this.$links.updateImmediately = (payload) => {
return apiWorkPackages.createWorkPackage(payload);
};
if (this.parent) {
this.$source._links['parent'] = {
href: this.parent.href
};
} else if ($stateParams.parent_id) {
this.$source._links['parent'] = {
href: v3Path.wp({wp: $stateParams.parent_id})
};
}
}
/**
* Exclude the schema _link from the linkable Resources.
*/
public $linkableKeys():string[] {
return _.without(super.$linkableKeys(), 'schema');
}
/**
* Get the current schema, assuming it is either:
* 1. Overridden by the current loaded form
* 2. Available as a schema state
*
* If it is neither, an exception is raised.
*/
public get schema():SchemaResource {
if (this.hasOverriddenSchema) {
return this.overriddenSchema!;
}
const state = schemaCacheService.state(this);
if (!state.hasValue()) {
throw `Accessing schema of ${this.id} without it being loaded.`;
}
return state.value!;
}
public get hasOverriddenSchema():boolean {
return this.overriddenSchema != null;
}
}
export interface WorkPackageResourceInterface extends WorkPackageResourceLinks, WorkPackageResourceEmbedded, WorkPackageResource {
}
function wpResource(...args:any[]) {
[
$q,
$stateParams,
$timeout,
I18n,
states,
apiWorkPackages,
wpCacheService,
wpCreate,
schemaCacheService,
NotificationsService,
wpNotificationsService,
AttachmentCollectionResource,
v3Path] = args;
return WorkPackageResource;
}
wpResource.$inject = [
'$q',
'$stateParams',
'$timeout',
'I18n',
'states',
'apiWorkPackages',
'wpCacheService',
'wpCreate',
'schemaCacheService',
'NotificationsService',
'wpNotificationsService',
'AttachmentCollectionResource',
'v3Path'
];
opApiModule.factory('WorkPackageResource', wpResource);