mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
41035: Fix for ALF-15225 - qt.length not performing as expected in search.lib.js 41047: RUSSIAN: Translation updates based on EN r40961 41049: GERMAN: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41050: SPANISH: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41051: FRENCH: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41052: ITALIAN: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41053: JAPANESE: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41054: DUTCH: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41055: RUSSIAN: Translation updates based on EN r41033. Fixes ALF-15749 and ALF-15720. 41061: ALF-11214 - IMAP subsystem is not successfully restarted after incorrect modification of IMAP properties via Admin Console 41063: RUSSIAN: Translation updates based on EN r41033 (encoding fixes) 41064: CHINESE: Translation updates based on EN r41033 41073: ALF-15760: Merged V4.0 to V4.1-BUG-FIX (another lost meta-inf revision) 34416: ALF-12992: Updated weblogic DD for SOLR 41074: ALF-15419 / ALF-14438: Merged V3.4-BUG-FIX to V4.1-BUG-FIX (PARTIAL) 37373: Merged PATCHES/V3.4.6 to V3.4-BUG-FIX 36821: ALF-13827 / ALF-14402: Make replicated caches recover from temporary comms failures by flushing when a change in peers is detected - We do not flush caches who replicate via copy (e.g. tickets cache) as these may not be recoverable 37122: ALF-13919 / ALF-14403: Merged DEV to PATCHES/V3.4.6 - Rework of Dmitry's implementation - Uses dynamic HQL query to retrieve JBPM workflow instances by specified query criteria - WorkflowInstancesGet web script no longer has to iterate over every workflow instance in the database! - DB index added to enable efficient querying by string variable - Hibernate tastic! 37188: ALF-13919 / ALF-14403: Worked around HQL polymorphism issues by using explicit variable subclass names in from clause 37204: ALF-13919 / ALF-14403: Fix to date range handling by Dmitry 41077: ALF-10730: Fixed object finder drop-down tree to show parent icon type when icon type not available 41089: ALF-13998: 'No items' error is highlighted in red, even that is not sever error. 41109: Part fix for ALF-11297 Disabled test: test-system-build-test has been removed from the test target dependency list 41118: Fixes: ALF-15765 and related issue ALF-15476: - Corrects merge of r34405 and r40159 - Adds people api to cloud proxy config - Cloud/Global Folder picker sites list now universally shows only those sites the user is a member of. 41119: ALF-15419 / ALF-14438: Merged DEV to V4.1-BUG-FIX 41117: ALF-15419 : CLONE Remove JBPM indexes present from upgrades The schema reference files were updated to contain JBPM_VARIABLEINSTANCE.IDX_VARINST_STRVAL index. 41130: ALF-15590: FSTR transfer of custom content type with parent cm:content fails - Fix by Dmitry 41131: ALF-15714: Bitrock Uninstaller: Uninstaller doesn't remove all files and folders in *nix - Fix provided by Bitrock 41132: Fix for ALF-14388 - Edit Online option is not supported for '.docm', 'dotm', '.xlsm' files - corrected mimetypes to be lower-case 41150: ALF-13287 Added the handling of UTC represented by "Z" within comparator 41154: Check for authentication errors when validating a ticket, if an error occurs re-authenticate. ALF-15394 41155: ALF-15569: User with '@' symbol in username cannot invite - note: since Ent 4.0.2(+) by default MT is pre-configured but not enabled 41173: Merged V4.1 to V4.1-BUG-FIX 41121: Merged BRANCHES/DEV/FEATURES/CLOUD1_CLOUDSYNC to BRANCHES/V4.1: 41003: CloudSync: ALF-15734 - force unsync (of last SSMN) on target causes repeating pull errors to appear in both logs 41026: CloudSync: ALF-15734 - force unsync (of last SSMN) on target causes repeating pull errors to appear in both logs 41039: CloudSync: ALF-15734 - force unsync (of last SSMN) on target causes repeating pull errors to appear in both logs 41086: CloudSync: ALF-15734 - force unsync (of last SSMN) on target causes repeating pull errors to appear in both logs 41123: Merged BRANCHES/DEV/FEATURES/CLOUD1_CLOUDSYNC to BRANCHES/V4.1: 41115: CloudSync: ALF-15734 - force unsync (of last SSMN) on target causes repeating pull errors to appear in both logs 41176: Attempt to debug unit test failure 41181: Store leak in AVMServiceTest.test_ETWOTWO_570() causing unit test failure 41184: ALF-15610: Copy Thai analyzer settings to its many SOLR locations 41194: ALF-11297: re-enable system build tests 41195: ALF-11297 ALF-15807: update activities system build tests after correction of ALF-4832 41201: Fix for ALF-15767 Group query using cm:authorityName 41202: Additional unit tests related to ALF-15731 TYPE:"..." queries no longer work for Lucene on 4.X 41203: Part 1 for ALF-15811 SOLR query increases DocBitSet inefficiently - check it makes any difference 41204: Merged BRANCHES/DEV/BELARUS/V4.1-BUG-FIX-2012_08_15 to BRANCHES/DEV/V4.1-BUG-FIX: (note: merging as-is ... refactor + unit test fix to follow in next commit) 40926: ALF-12586: Admin Console shows usage as zero - if user is deleted and then re-created (eg. re-synchronized via LDAP or manually) 40974: ALF-12586: Admin Console shows usage as zero - if user is deleted and then re-created (eg. re-synchronized via LDAP or manually) 41205: ALF-12586: Admin Console shows usage as zero - if user is deleted and then re-created (eg. re-synchronized via LDAP or manually) - review and refactor the proposed/merged fix (see previous commit) - fix unit test so that it accounts for previous content (as per the original issue) - also: add missing test to suite and fix that test to work with the new fix (ie. cleanup previous content, else need to account for it) 41210: Part 2: ALF-14861 SOLR to scale for non-admin users in 100k sites and a subgroup of each of 1000 independent groupings with 1000 subgroups - do not expand authorities for thoses with the ADMINISTRATOR_ROLE as they can read all anyway 41216: ALF-11297: system build tests need a database cleanup before running 41222: ALF-15740, ALF-14744: Update rule firing broken for content created in Explorer - Old code lurking around that used to use the inline editable aspect to detect events handled by the CreateNodeRuleTrigger was removed from OnContentUpdateRuleTrigger and replaced with a check for ASPECT_NO_CONTENT 41223: Added missing swf.languagedir setting to enterprise alfresco-global.properties 41230: GERMAN: Translation updates based on EN rev41099. 41232: SPANISH: Translation updates based on EN rev41099. 41233: FRENCH: Translation updates based on EN rev41099. 41234: ITALIAN: Translation updates based on EN rev41099. 41235: JAPANESE: Translation updates based on EN rev41099. 41236: DUTCH: Translation updates based on EN rev41099. 41237: RUSSIAN: Translation updates based on EN rev41099. 41239: CHINESE: Translation updates based on EN rev41099. 41254: ALF-15628: Avoid edit online (SPP, WRITE_LOCK) clashing with edit offline (CheckOutCheckInService, READ_ONLY_LOCK) - Rationalization of work by Alex Malinovsky - WebDAVMethod.checkNode() now properly checks whether nodes without WebDAV lock info are writeable - CheckOutCheckInService won't allow checkout of a node with an existing WRITE_LOCK by the same user - they must unlock first - Propagation of correct status codes 41264: ALF-15628: Fix CheckOutCheckInService test failures 41265: ALF-15699: Reverse merged the following, thus downgrading us back to swftools 0.9.1 40208: ALF-12831: Upgrade to swftools 0.9.2 41266: Rush'n in some translation updates from Gloria 41267: ALF-15628: Fix compilation problem 41269: Merged V3.4-BUG-FIX to V4.1-BUG-FIX (RECORD ONLY) 41224: ALF-14856: Merged V4.1-BUG-FIX to V3.4-BUG-FIX 41268: ALF-15459: Merged PATCHES/V4.0.2 to V3.4-BUG-FIX Merged V4.1-BUG-FIX to V3.4-BUG-FIX 41274: ALF-15608: Merged V3.4-BUG-FIX to V4.1-BUG-FIX 41272: ALF-15567: Allow links to be followed through WebDAV on port 80 using basic auth on XP 41277: ALF-12586: Admin Console shows usage as zero - if user is deleted and then re-created (eg. re-synchronized via LDAP or manually) - fix test fallout after merge/fix (note: failed for PostgreSQL but not MySQL - although fix was not DB-specific) 41278: ALF-15840 Error logged when "No thumbnail present in file" even though this is normal 41284: ALF-14875: Serialize direct permissions after inherited permissions so that they take precedence in any lookups in permission dialogs 41290: JAPANESE: Translation updates based on EN r41099 Fixes: ALF-14565 41296: ALF-15251 CIFS: Checked out document is not marked as locked in CIFS 41299: ALF-15714: Bitrock Uninstaller: Uninstaller doesn't remove all files and folders in *nix - Additional fix provided by Bitrock 41303: Fix for ALF-15799 Under high concurrency load balanced Solr throws an Antlr related NPE - do not skip IO Exceptions 41306: Incremented version revision for 4.1.2 41309: ALF-15827: Added FORMACTION, FORMMETHOD and ACTION HTML attributes to grey list to close security hole (updated Surf libs r1136) 41318: ALF-15857: Lucene FTS indexer opens streams to all documents to be indexed in a transaction simultaneously - Now stream opening is delayed until the point where the document is being written to the index 41322: Fix for ALF-15858 SOLR ACL tracking can stall or miss acls during tracking 41323: Chemistry client java to create test data for ALF-15858, ALF-15782, CLOUD-596, ALF-15753 etc 41326: ALF-15234: IE 8 or IE9 Download .pps as .ppt -Switched the order of the filename headers to better support non-conforming browsers (rfc 5987). 41330: ALF-14875: Reversed r41284 because it didn't solve the problem in the UI. Kev reviewing. 41332: ALF-10688: Can't deactivate an account when alfrescoNtlm follows another authentication subsystem in the authentication chain - Now, if a account is known to be 'mutable' then the enabled flag is read from the mutable authentication service 41337: Merged V3.4-BUG-FIX (3.4.11) to V4.1-BUG-FIX (4.1.2) 41336: TransformerDebug: Use debug rather than trace when there are transformers but they are all unavailable. 41339: ALF-15840 Error logged when "No thumbnail present in file" even though this is normal - Found some more cases where this is logged as an ERROR 41342: ALF-11087 (Missing icon file: components\images\filetypes\generic-tag-32.png) 41344: ALF-15863 (* search values): Merged HEAD to V4.1-BUG-FIX (4.1.2) 40849: ALF-12839 "Share - Inconsistency in adding a user or a group into a group" part 2 - Making the users console stop "*" searches, just like the groups console when the min search length is set to larger than 0. 41346: ALF-15237 - REST API Group children lists username for fullName and displayName 41350: Merge V3.4-BUG-FIX to V4.1-BUG-FIX: 41065: Disconnect existing CIFS sessions from the same client when a virtual circuit zero session is opened. ALF-13815 41280: Moved session cleanup config into the base authenticator, added support to passthru/base authentication. ALF-13815 41351: Ported database filesystem changes to fix session disconnect, from V3.4. 41352: Merge V3.4-BUG-FIX to V4.1-BUG-FIX: 41067: Added session disconnect support to the Alfresco CIFS authenticator. ALF-13815. 41281: Added session cleanup support to passthru authenticator, session cleanup config moved to base class. ALF-13815 41353: Update svn:mergeinfo 41355: Fix for ALF-15869 - "Site Content" dashlet shows all documents from all the sites in Alfresco Share - mistakenly did a record-only merge of this from 4.1->4.1.1 41363: Fix for ALF-14875 - Manage permissions shows the permission 'No privileges' for All Other Users - reworked the permissions dialog and permissions panels to correctly handle multiple permissions on a special permissions group such as GROUP_EVERYONE - now correctly gets/sets permissions for GROUP_EVERYONE - this also fixes ALF-12014 - in that it allows custom SiteXYZ permissions to work correctly again also (will need manual backport for 3.4.X though) - removed hacks related to previous attempts to fix the above issue - added lots of comments around relevant sections to add in future refactoring or understanding 41371: Merged BRANCHES/DEV/V3.4-BUG-FIX to BRANCHES/DEV/V4.1-BUG-FIX (RECORD ONLY) 41370: Fix for ALF-12014 - Share - Custom role causes incorrect display of permissions Manual merge of changes to 4.1.2 41399: ALF-13438: java.lang.OutOfMemoryError appears in alfresco log when trying to create few thousands of sites consistently. - The v4.0 parent assocs cache used to store every verion of every node's parent assocs, so as we added a node to 60,000 sites, we retained all previous versions of the user's parent association map - After creating 10,000 sites, the cache size was about 4GB, containing about 2 million parents - Now we use a specialized class that allows an upper limit to be set on the total number of cached parents as well as children. - Because the cache is keyed by node transaction ID, the cache can be non-clustered and non-transactional - Once the average number of parents is more than 8, the cache will drop its oldest entries - ParentAssocsInfo also now uses a compact TreeMap instead of a HashMap 41401: ALF-13438: Small correction 41406: ALF-13438: Fix failing unit tests - Removed unused node.parentAssocsSharedCache and corrected node.childByNameCache not to reference it! - Because parent assocs are cached by transaction ID, we must always invalidate them on an in-transaction version increment 41409: Logging of unexepected errors on FTP 41411: Fix possible FTP data session leak if client mixes PORT and PASV commands. ALF-15126 41412: ALF-15845 : Clone for Hotfix: Word document on Windows via CIFS becomes locked (Read Only) when network drops temporarily 41415: Fixes: ALF-15649: Removes country locale from files with it hard coded. 41419: ALF-14599: Removed ftp.ipv6.enabled from enterprise overlay and bundles 41426: ALF-15845 Clone for Hotfix: Word document on Windows via CIFS becomes locked (Read Only) when network drops temporarily Roll back changes to DiskDriver interface in favour of hacking NetworkFile. 41440: JAPANESE: Translation update based on EN r41099 41446: ALF-13091: Remove unecessary bean post processors from sub ssytem context and remove CXF's Jsr250BeanPostProcessor. 41458: RUSSIAN: Further translation updates following linguistic review. 41459: ALF-15897: Revert revision 41446, an attempted fix for ALF-13091 41487: Fix for ALF-15910 SOLR - Add index warming and filter pointless entries from the filter cache Fix for ALF-15851 Too many live instances of SolrIndexSearcher at one time resulting in OOM - Alfresco 4.1.1 - build 151 41506: Merged DEV to V4.1-BUG-FIX 41505: ALF-15879: PostgreSQL: upgrade 2.2.8 (577) -> 3.4.10 (703) -> 4.1.1 (159) failed. - Make dropping "store_id" index and "alf_node_store_id_key" constraint optional in 4.1.1 upgrade script because clean 3.4 has "store_id" index and doesn't have "alf_node_store_id_key" constraint, but 3.4 upgraded from 2.2 has "alf_node_store_id_key" constraint and doesn't have "store_id" index. 41531: Fixed ALF-15687, so that any user (except for Admin) won’t be able to retrieve any other user’s preferences via REST API. Also, updated the preferences controllers for the Post and Delete. 41539: ALF-15899: Inbound email does not support multiple recipient folders - Fix by Dmitry Vaserin 41540: Merged V3.4-BUG-FIX to V4.1-BUG-FIX 40794: Merged DEV to V3.4-BUG-FIX 40793: ALF-13752 Saving Word (mac 2011) documents via CIFS into a folder with Versionable rules on Mac OS X Lion (Fix for 3.4) In ContentDiskDriver.renameFile() was added a check whether a node in the archive. 40806: Fix for ALF-9787 - Hiding sites in Share with permissionsDefinitions.xml [creates a permissions error in the blog portion of the site] 40922: Merged DEV to V3.4-BUG-FIX (reviewed by Frederik) 40488: ALF-13357 : Empty outcome when a timer is invoked A check for transitionName was added to AlfrescoTimer to support custom transitions. 40940: ALF-15696: Remove svnkit.jar - makes the build fail if the version of installed command line svn is 1.7. Using commandline binding for <svn> Ant task to be consistent with other parts of the build. 41066: Various fixes to the database filesystem for session disconnect cleanup. 41068: Updates to the Alfresco filesystem for session cleanup. ALF-13815 41301: Merged PATCHES/V3.4.9 to V3.4-BUG-FIX 40966: ALF-15846 / ALF-15709: OOM on cascading reindex - Avoid buffering of all the affected PATH documents in memory - used the set of 'visited' paths to delay generation to the final flush. 41044: ALF-15847 / ALF-15748: Lucene indexer can make sub-optimal cascade reindex decisions during an LDAP sync. - When a user in 6 groups was removed from a massive group, the massive group was getting cascade reindexed rather than the user - Logic adjusted as follows: For nodes with 5 or less parents, we always cascade reindex the child node. For nodes with more than 5 parents, we cascade reindex the parent node if it has less children than the child has parents. 41395: ALF-15715: Unable to edit properties whilst transformation in progress - Delayed all changes that would potentially lock the parent node row while the thumbnail is generating. - Reorganized RenditionContext to lazily instantiate its destination node. - Changed render destination to use temporary ContentWriter until render is complete. 41396: ALF-15715: Fix for failing unit tests. 41413: Fix for issue where user calendar remote api was generating invalid date searches for user dashlet calendar. 41509: Merged DEV to V3.4-BUG-FIX 41507: ALF-12833: Issues installing Alfresco on WebSphere when the server doesn't have internet access Context-param which Sets "http://apache.org/xml/features/nonvalidating/load-external-dtd" feature on the SAXParser to false if this parameter is false 41510: ALF-15171: After addition of a secondary parent association to a container, not all index paths were being regenerated due to a logic error 41512: ALF-15919: Merged PATCHES/V3.4.10 to V3.4-BUG-FIX 41091: ALF-15723: Merged DEV to PATCHES/V3.4.10 26579: Switch the transformer to use Tika git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@41543 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
3878 lines
143 KiB
Java
3878 lines
143 KiB
Java
/*
|
|
* Copyright (C) 2005-2012 Alfresco Software Limited.
|
|
*
|
|
* This file is part of Alfresco
|
|
*
|
|
* Alfresco is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Alfresco 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 Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package org.alfresco.repo.jscript;
|
|
|
|
import java.io.BufferedInputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.io.InputStreamReader;
|
|
import java.io.OutputStream;
|
|
import java.io.Reader;
|
|
import java.io.Serializable;
|
|
import java.nio.charset.Charset;
|
|
import java.text.Collator;
|
|
import java.text.MessageFormat;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.Collection;
|
|
import java.util.Comparator;
|
|
import java.util.Date;
|
|
import java.util.HashMap;
|
|
import java.util.HashSet;
|
|
import java.util.LinkedHashMap;
|
|
import java.util.LinkedHashSet;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
import java.util.StringTokenizer;
|
|
|
|
import org.alfresco.error.AlfrescoRuntimeException;
|
|
import org.alfresco.model.ApplicationModel;
|
|
import org.alfresco.model.ContentModel;
|
|
import org.alfresco.query.PagingRequest;
|
|
import org.alfresco.query.PagingResults;
|
|
import org.alfresco.repo.action.executer.TransformActionExecuter;
|
|
import org.alfresco.repo.content.transform.UnimportantTransformException;
|
|
import org.alfresco.repo.content.transform.magick.ImageTransformationOptions;
|
|
import org.alfresco.repo.model.filefolder.FileFolderServiceImpl.InvalidTypeException;
|
|
import org.alfresco.repo.search.QueryParameterDefImpl;
|
|
import org.alfresco.repo.security.authentication.AuthenticationUtil;
|
|
import org.alfresco.repo.security.authentication.AuthenticationUtil.RunAsWork;
|
|
import org.alfresco.repo.tagging.script.TagScope;
|
|
import org.alfresco.repo.thumbnail.ThumbnailDefinition;
|
|
import org.alfresco.repo.thumbnail.ThumbnailHelper;
|
|
import org.alfresco.repo.thumbnail.ThumbnailRegistry;
|
|
import org.alfresco.repo.thumbnail.script.ScriptThumbnail;
|
|
import org.alfresco.repo.transaction.AlfrescoTransactionSupport;
|
|
import org.alfresco.repo.version.VersionModel;
|
|
import org.alfresco.repo.workflow.jscript.JscriptWorkflowInstance;
|
|
import org.alfresco.scripts.ScriptException;
|
|
import org.alfresco.service.ServiceRegistry;
|
|
import org.alfresco.service.cmr.action.Action;
|
|
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
|
|
import org.alfresco.service.cmr.dictionary.DictionaryService;
|
|
import org.alfresco.service.cmr.lock.LockStatus;
|
|
import org.alfresco.service.cmr.model.FileFolderService;
|
|
import org.alfresco.service.cmr.model.FileInfo;
|
|
import org.alfresco.service.cmr.model.FileNotFoundException;
|
|
import org.alfresco.service.cmr.repository.AssociationRef;
|
|
import org.alfresco.service.cmr.repository.ChildAssociationRef;
|
|
import org.alfresco.service.cmr.repository.ContentData;
|
|
import org.alfresco.service.cmr.repository.ContentReader;
|
|
import org.alfresco.service.cmr.repository.ContentService;
|
|
import org.alfresco.service.cmr.repository.ContentWriter;
|
|
import org.alfresco.service.cmr.repository.NoTransformerException;
|
|
import org.alfresco.service.cmr.repository.NodeRef;
|
|
import org.alfresco.service.cmr.repository.NodeService;
|
|
import org.alfresco.service.cmr.repository.Path;
|
|
import org.alfresco.service.cmr.repository.StoreRef;
|
|
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
|
import org.alfresco.service.cmr.repository.TransformationOptions;
|
|
import org.alfresco.service.cmr.repository.datatype.DefaultTypeConverter;
|
|
import org.alfresco.service.cmr.search.QueryParameterDefinition;
|
|
import org.alfresco.service.cmr.security.AccessPermission;
|
|
import org.alfresco.service.cmr.security.AccessStatus;
|
|
import org.alfresco.service.cmr.security.PermissionService;
|
|
import org.alfresco.service.cmr.thumbnail.ThumbnailService;
|
|
import org.alfresco.service.cmr.version.Version;
|
|
import org.alfresco.service.cmr.version.VersionHistory;
|
|
import org.alfresco.service.cmr.version.VersionType;
|
|
import org.alfresco.service.cmr.workflow.WorkflowInstance;
|
|
import org.alfresco.service.cmr.workflow.WorkflowService;
|
|
import org.alfresco.service.namespace.NamespaceException;
|
|
import org.alfresco.service.namespace.NamespacePrefixResolver;
|
|
import org.alfresco.service.namespace.NamespacePrefixResolverProvider;
|
|
import org.alfresco.service.namespace.NamespaceService;
|
|
import org.alfresco.service.namespace.QName;
|
|
import org.alfresco.service.namespace.RegexQNamePattern;
|
|
import org.alfresco.util.FileFilterMode;
|
|
import org.alfresco.util.GUID;
|
|
import org.alfresco.util.ISO8601DateFormat;
|
|
import org.alfresco.util.ISO9075;
|
|
import org.alfresco.util.Pair;
|
|
import org.alfresco.util.FileFilterMode.Client;
|
|
import org.apache.commons.logging.Log;
|
|
import org.apache.commons.logging.LogFactory;
|
|
import org.json.JSONException;
|
|
import org.json.JSONObject;
|
|
import org.mozilla.javascript.Context;
|
|
import org.mozilla.javascript.Scriptable;
|
|
import org.mozilla.javascript.ScriptableObject;
|
|
import org.mozilla.javascript.UniqueTag;
|
|
import org.mozilla.javascript.Wrapper;
|
|
import org.springframework.extensions.surf.util.Content;
|
|
import org.springframework.extensions.surf.util.I18NUtil;
|
|
import org.springframework.extensions.surf.util.ParameterCheck;
|
|
import org.springframework.extensions.surf.util.URLEncoder;
|
|
|
|
/**
|
|
* Script Node class implementation, specific for use by ScriptService as part of the object model.
|
|
* <p>
|
|
* The class exposes Node properties, children and assocs as dynamically populated maps and lists. The various collection classes are mirrored as JavaScript properties. So can be
|
|
* accessed using standard JavaScript property syntax, such as <code>node.children[0].properties.name</code>.
|
|
* <p>
|
|
* Various helper methods are provided to access common and useful node variables such as the content url and type information.
|
|
*
|
|
* @author Kevin Roast
|
|
*/
|
|
public class ScriptNode implements Scopeable, NamespacePrefixResolverProvider
|
|
{
|
|
private static final long serialVersionUID = -3378946227712939601L;
|
|
|
|
private static Log logger = LogFactory.getLog(ScriptNode.class);
|
|
|
|
private final static String NAMESPACE_BEGIN = "" + QName.NAMESPACE_BEGIN;
|
|
|
|
private final static String CONTENT_DEFAULT_URL = "/d/d/{0}/{1}/{2}/{3}";
|
|
private final static String CONTENT_DOWNLOAD_URL = "/d/a/{0}/{1}/{2}/{3}";
|
|
private final static String CONTENT_PROP_URL = "/d/d/{0}/{1}/{2}/{3}?property={4}";
|
|
private final static String CONTENT_DOWNLOAD_PROP_URL = "/d/a/{0}/{1}/{2}/{3}?property={4}";
|
|
private final static String FOLDER_BROWSE_URL = "/n/browse/{0}/{1}/{2}";
|
|
|
|
/** Root scope for this object */
|
|
protected Scriptable scope;
|
|
|
|
/** Node Value Converter */
|
|
protected NodeValueConverter converter = null;
|
|
|
|
/** Cached values */
|
|
protected NodeRef nodeRef;
|
|
|
|
private FileInfo nodeInfo;
|
|
|
|
private String name;
|
|
private QName type;
|
|
protected String id;
|
|
protected String siteName;
|
|
protected boolean siteNameResolved = false;
|
|
|
|
/** The aspects applied to this node */
|
|
protected Set<QName> aspects = null;
|
|
|
|
/** The target associations from this node */
|
|
private ScriptableQNameMap<String, Object> targetAssocs = null;
|
|
|
|
/** The source associations to this node */
|
|
private ScriptableQNameMap<String, Object> sourceAssocs = null;
|
|
|
|
/** The child associations for this node */
|
|
private ScriptableQNameMap<String, Object> childAssocs = null;
|
|
|
|
/** The children of this node */
|
|
private Scriptable children = null;
|
|
|
|
/** The properties of this node */
|
|
private ScriptableQNameMap<String, Serializable> properties = null;
|
|
|
|
/** The versions of this node */
|
|
private Scriptable versions = null;
|
|
|
|
/** The active workflows acting on this node */
|
|
private Scriptable activeWorkflows = null;
|
|
|
|
protected ServiceRegistry services = null;
|
|
private NodeService nodeService = null;
|
|
private FileFolderService fileFolderService = null;
|
|
private Boolean isDocument = null;
|
|
private Boolean isContainer = null;
|
|
private Boolean isLinkToDocument = null;
|
|
private Boolean isLinkToContainer = null;
|
|
private Boolean hasChildren = null;
|
|
private String displayPath = null;
|
|
protected TemplateImageResolver imageResolver = null;
|
|
protected ScriptNode parent = null;
|
|
private ChildAssociationRef primaryParentAssoc = null;
|
|
private ScriptableQNameMap<String, Object> parentAssocs = null;
|
|
// NOTE: see the reset() method when adding new cached members!
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Construction
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param nodeRef The NodeRef this Node wrapper represents
|
|
* @param services The ServiceRegistry the Node can use to access services
|
|
*/
|
|
public ScriptNode(NodeRef nodeRef, ServiceRegistry services)
|
|
{
|
|
this(nodeRef, services, null);
|
|
}
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param nodeInfo The FileInfo this Node wrapper represents
|
|
* @param services The ServiceRegistry the Node can use to access services
|
|
* @param scope Root scope for this Node
|
|
*/
|
|
public ScriptNode(FileInfo nodeInfo, ServiceRegistry services, Scriptable scope)
|
|
{
|
|
this(nodeInfo.getNodeRef(), services, scope);
|
|
|
|
this.nodeInfo = nodeInfo;
|
|
}
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param nodeRef The NodeRef this Node wrapper represents
|
|
* @param services The ServiceRegistry the Node can use to access services
|
|
* @param scope Root scope for this Node
|
|
*/
|
|
public ScriptNode(NodeRef nodeRef, ServiceRegistry services, Scriptable scope)
|
|
{
|
|
if (nodeRef == null)
|
|
{
|
|
throw new IllegalArgumentException("NodeRef must be supplied.");
|
|
}
|
|
|
|
if (services == null)
|
|
{
|
|
throw new IllegalArgumentException("The ServiceRegistry must be supplied.");
|
|
}
|
|
|
|
this.nodeRef = nodeRef;
|
|
this.id = nodeRef.getId();
|
|
this.services = services;
|
|
this.nodeService = services.getNodeService();
|
|
this.fileFolderService = services.getFileFolderService();
|
|
this.scope = scope;
|
|
}
|
|
|
|
@Override
|
|
public int hashCode()
|
|
{
|
|
final int PRIME = 31;
|
|
int result = 1;
|
|
result = PRIME * result + ((nodeRef == null) ? 0 : nodeRef.hashCode());
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object obj)
|
|
{
|
|
if (this == obj) return true;
|
|
if (obj == null) return false;
|
|
if (getClass() != obj.getClass()) return false;
|
|
if (!nodeRef.equals(((ScriptNode)obj).nodeRef)) return false;
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Factory method
|
|
*/
|
|
public ScriptNode newInstance(NodeRef nodeRef, ServiceRegistry services, Scriptable scope)
|
|
{
|
|
return new ScriptNode(nodeRef, services, scope);
|
|
}
|
|
|
|
public ScriptNode newInstance(FileInfo nodeInfo, ServiceRegistry services, Scriptable scope)
|
|
{
|
|
return new ScriptNode(nodeInfo, services, scope);
|
|
}
|
|
|
|
/**
|
|
* @see org.alfresco.repo.jscript.Scopeable#setScope(org.mozilla.javascript.Scriptable)
|
|
*/
|
|
public void setScope(Scriptable scope)
|
|
{
|
|
this.scope = scope;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Node Wrapper API
|
|
|
|
/**
|
|
* @return The GUID for the node
|
|
*/
|
|
public String getId()
|
|
{
|
|
return this.id;
|
|
}
|
|
|
|
/**
|
|
* @return the store type for the node
|
|
*/
|
|
public String getStoreType()
|
|
{
|
|
return this.nodeRef.getStoreRef().getProtocol();
|
|
}
|
|
|
|
/**
|
|
* @return the store id for the node
|
|
*/
|
|
public String getStoreId()
|
|
{
|
|
return this.nodeRef.getStoreRef().getIdentifier();
|
|
}
|
|
|
|
/**
|
|
* @return Returns the NodeRef this Node object represents
|
|
*/
|
|
public NodeRef getNodeRef()
|
|
{
|
|
return this.nodeRef;
|
|
}
|
|
|
|
/**
|
|
* @return Returns the QName type.
|
|
*/
|
|
public QName getQNameType()
|
|
{
|
|
if (this.type == null)
|
|
{
|
|
this.type = this.nodeService.getType(this.nodeRef);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
/**
|
|
* @return Returns the type.
|
|
*/
|
|
public String getType()
|
|
{
|
|
return getQNameType().toString();
|
|
}
|
|
|
|
/**
|
|
* @return Returns the type in short format.
|
|
*/
|
|
public String getTypeShort()
|
|
{
|
|
return this.getShortQName(getQNameType());
|
|
}
|
|
|
|
/**
|
|
* @return Helper to return the 'name' property for the node
|
|
*/
|
|
public String getName()
|
|
{
|
|
if (this.name == null)
|
|
{
|
|
// try and get the name from the properties first
|
|
this.name = (String) getProperties().get("cm:name");
|
|
|
|
// if we didn't find it as a property get the name from the association name
|
|
if (this.name == null)
|
|
{
|
|
ChildAssociationRef parentRef = this.nodeService.getPrimaryParent(this.nodeRef);
|
|
if (parentRef != null && parentRef.getQName() != null)
|
|
{
|
|
this.name = parentRef.getQName().getLocalName();
|
|
}
|
|
else
|
|
{
|
|
this.name = "";
|
|
}
|
|
}
|
|
}
|
|
|
|
return this.name;
|
|
}
|
|
|
|
/**
|
|
* Helper to set the 'name' property for the node.
|
|
*
|
|
* @param name Name to set
|
|
*/
|
|
public void setName(String name)
|
|
{
|
|
if (name != null)
|
|
{
|
|
QName typeQName = getQNameType();
|
|
if ((services.getDictionaryService().isSubClass(typeQName, ContentModel.TYPE_FOLDER) &&
|
|
!services.getDictionaryService().isSubClass(typeQName, ContentModel.TYPE_SYSTEM_FOLDER)) ||
|
|
services.getDictionaryService().isSubClass(typeQName, ContentModel.TYPE_CONTENT))
|
|
{
|
|
try
|
|
{
|
|
this.services.getFileFolderService().rename(this.nodeRef, name);
|
|
}
|
|
catch (FileNotFoundException e)
|
|
{
|
|
throw new AlfrescoRuntimeException("Failed to rename node " + nodeRef + " to " + name, e);
|
|
}
|
|
}
|
|
this.getProperties().put(ContentModel.PROP_NAME.toString(), name.toString());
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return The children of this Node as JavaScript array of Node object wrappers
|
|
*/
|
|
public Scriptable getChildren()
|
|
{
|
|
if (this.children == null)
|
|
{
|
|
List<ChildAssociationRef> childRefs = this.nodeService.getChildAssocs(this.nodeRef);
|
|
Object[] children = new Object[childRefs.size()];
|
|
for (int i = 0; i < childRefs.size(); i++)
|
|
{
|
|
// create our Node representation from the NodeRef
|
|
children[i] = newInstance(childRefs.get(i).getChildRef(), this.services, this.scope);
|
|
}
|
|
|
|
// Do a locale-sensitive sort by name
|
|
sort(children);
|
|
|
|
this.children = Context.getCurrentContext().newArray(this.scope, children);
|
|
this.hasChildren = (children.length != 0);
|
|
}
|
|
|
|
return this.children;
|
|
}
|
|
|
|
/**
|
|
* Performs a locale-sensitive sort by name of a node array
|
|
* @param nodes the node array
|
|
*/
|
|
private static void sort(Object[] nodes)
|
|
{
|
|
final Collator col = Collator.getInstance(I18NUtil.getLocale());
|
|
Arrays.sort(nodes, new Comparator<Object>(){
|
|
@Override
|
|
public int compare(Object o1, Object o2)
|
|
{
|
|
return col.compare(((ScriptNode)o1).getName(), ((ScriptNode)o2).getName());
|
|
}});
|
|
}
|
|
|
|
/**
|
|
* @return true if the Node has children
|
|
*/
|
|
public boolean getHasChildren()
|
|
{
|
|
if (this.hasChildren == null)
|
|
{
|
|
this.hasChildren = !this.services.getNodeService().getChildAssocs(
|
|
getNodeRef(), RegexQNamePattern.MATCH_ALL, RegexQNamePattern.MATCH_ALL, false).isEmpty();
|
|
}
|
|
return hasChildren;
|
|
}
|
|
|
|
/**
|
|
* childByNamePath returns the Node at the specified 'cm:name' based Path walking the children of this Node.
|
|
* So a valid call might be:
|
|
* <code>mynode.childByNamePath("/QA/Testing/Docs");</code>
|
|
*
|
|
* @return The ScriptNode or null if the node is not found.
|
|
*/
|
|
public ScriptNode childByNamePath(String path)
|
|
{
|
|
ScriptNode child = null;
|
|
|
|
if (this.services.getDictionaryService().isSubClass(getQNameType(), ContentModel.TYPE_FOLDER))
|
|
{
|
|
// The current node is a folder e.g. Company Home and standard child folders.
|
|
// optimized code path for cm:folder and sub-types supporting getChildrenByName() method
|
|
NodeRef result = null;
|
|
final StringTokenizer t = new StringTokenizer(path, "/");
|
|
if (t.hasMoreTokens())
|
|
{
|
|
result = this.nodeRef;
|
|
while (t.hasMoreTokens() && result != null)
|
|
{
|
|
final String name = t.nextToken();
|
|
final NodeRef lastName = result;
|
|
result = AuthenticationUtil.runAsSystem(new RunAsWork<NodeRef>()
|
|
{
|
|
@Override
|
|
public NodeRef doWork() throws Exception
|
|
{
|
|
NodeRef child = null;
|
|
if (t.hasMoreTokens())
|
|
{
|
|
// allow traversal of a cm:name based path even if user cannot retrieve the node directly
|
|
child = nodeService.getChildByName(lastName, ContentModel.ASSOC_CONTAINS, name);
|
|
}
|
|
else
|
|
{
|
|
// final node must be accessible to the user via the usual ACL permission checks
|
|
child = nodeService.getChildByName(lastName, ContentModel.ASSOC_CONTAINS, name);
|
|
if (child != null && AccessStatus.ALLOWED != services.getPermissionService().hasPermission(child, PermissionService.READ_PROPERTIES))
|
|
{
|
|
child = null;
|
|
}
|
|
}
|
|
return child;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
child = (result != null ? newInstance(result, this.services, this.scope) : null);
|
|
}
|
|
else
|
|
{
|
|
// The current node is not a folder. It does not support the cm:contains association.
|
|
// Convert the name based path to a valid XPath query
|
|
StringBuilder xpath = new StringBuilder(path.length() << 1);
|
|
StringTokenizer t = new StringTokenizer(path, "/");
|
|
int count = 0;
|
|
QueryParameterDefinition[] params = new QueryParameterDefinition[t.countTokens()];
|
|
DataTypeDefinition ddText =
|
|
this.services.getDictionaryService().getDataType(DataTypeDefinition.TEXT);
|
|
NamespaceService ns = this.services.getNamespaceService();
|
|
while (t.hasMoreTokens())
|
|
{
|
|
if (xpath.length() != 0)
|
|
{
|
|
xpath.append('/');
|
|
}
|
|
String strCount = Integer.toString(count);
|
|
xpath.append("*[@cm:name=$cm:name")
|
|
.append(strCount)
|
|
.append(']');
|
|
params[count++] = new QueryParameterDefImpl(
|
|
QName.createQName(NamespaceService.CONTENT_MODEL_PREFIX, "name" + strCount, ns),
|
|
ddText,
|
|
true,
|
|
t.nextToken());
|
|
}
|
|
|
|
Object[] nodes = getChildrenByXPath(xpath.toString(), params, true);
|
|
|
|
child = (nodes.length != 0) ? (ScriptNode)nodes[0] : null;
|
|
}
|
|
|
|
return child;
|
|
}
|
|
|
|
/**
|
|
* @return Returns a JavaScript array of Nodes at the specified XPath starting at this Node.
|
|
* So a valid call might be <code>mynode.childrenByXPath("*[@cm:name='Testing']/*");</code>
|
|
*/
|
|
public Scriptable childrenByXPath(String xpath)
|
|
{
|
|
return Context.getCurrentContext().newArray(this.scope, getChildrenByXPath(xpath, null, false));
|
|
}
|
|
|
|
/**
|
|
* @return Returns a JavaScript array of child file/folder nodes for this nodes.
|
|
* Automatically retrieves all sub-types of cm:content and cm:folder, also removes
|
|
* system folder types from the results.
|
|
* This is equivalent to @see FileFolderService.list()
|
|
*/
|
|
public Scriptable childFileFolders()
|
|
{
|
|
return childFileFolders(true, true, null);
|
|
}
|
|
|
|
/**
|
|
* @param files Return files extending from cm:content
|
|
* @param folders Return folders extending from cm:folder - ignoring sub-types of cm:systemfolder
|
|
*
|
|
* @return Returns a JavaScript array of child file/folder nodes for this nodes.
|
|
* Automatically retrieves all sub-types of cm:content and cm:folder, also removes
|
|
* system folder types from the results.
|
|
* This is equivalent to @see FileFolderService.listFiles() and @see FileFolderService.listFolders()
|
|
*/
|
|
public Scriptable childFileFolders(boolean files, boolean folders)
|
|
{
|
|
return childFileFolders(files, folders, null);
|
|
}
|
|
|
|
/**
|
|
* @param files Return files extending from cm:content
|
|
* @param folders Return folders extending from cm:folder - ignoring sub-types of cm:systemfolder
|
|
* @param ignoreTypes Also optionally removes additional type qnames. The additional type can be
|
|
* specified in short or long qname string form as a single string or an Array e.g. "fm:forum".
|
|
*
|
|
* @return Returns a JavaScript array of child file/folder nodes for this nodes.
|
|
* Automatically retrieves all sub-types of cm:content and cm:folder, also removes
|
|
* system folder types from the results.
|
|
* This is equivalent to @see FileFolderService.listFiles() and @see FileFolderService.listFolders()
|
|
*/
|
|
public Scriptable childFileFolders(boolean files, boolean folders, Object ignoreTypes)
|
|
{
|
|
return childFileFolders(files, folders, ignoreTypes, -1, -1, 0, null, null, null).getPage();
|
|
}
|
|
|
|
/**
|
|
* @param files Return files extending from cm:content
|
|
* @param folders Return folders extending from cm:folder - ignoring sub-types of cm:systemfolder
|
|
* @param ignoreTypes Also optionally removes additional type qnames. The additional type can be
|
|
* specified in short or long qname string form as a single string or an Array e.g. "fm:forum".
|
|
* @param maxItems Max number of items
|
|
*
|
|
* @return Returns ScriptPagingNodes which includes a JavaScript array of child file/folder nodes for this nodes.
|
|
* Automatically retrieves all sub-types of cm:content and cm:folder, also removes
|
|
* system folder types from the results.
|
|
* This is equivalent to @see FileFolderService.listFiles() and @see FileFolderService.listFolders()
|
|
*
|
|
* @deprecated API for review (subject to change prior to release)
|
|
*
|
|
* @author janv
|
|
* @since 4.0
|
|
*/
|
|
public ScriptPagingNodes childFileFolders(boolean files, boolean folders, Object ignoreTypes, int maxItems)
|
|
{
|
|
return childFileFolders(files, folders, ignoreTypes, 0, maxItems, 0, null, null, null);
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
/**
|
|
* @param files Return files extending from cm:content
|
|
* @param folders Return folders extending from cm:folder - ignoring sub-types of cm:systemfolder
|
|
* @param ignoreTypes Also optionally removes additional type qnames. The additional type can be
|
|
* specified in short or long qname string form as a single string or an Array e.g. "fm:forum".
|
|
* @param skipOffset Items to skip (e.g. 0 or (num pages to skip * size of page)
|
|
* @param maxItems Max number of items (eg. size of page)
|
|
* @param requestTotalCountMax Request total count (upto a given max total count)
|
|
* Note: if 0 then total count is not requested and the query may be able to optimise/cutoff for max items)
|
|
* @param sortProp Optional sort property as a prefix qname string (e.g. "cm:name"). Also supports special
|
|
* content case (i.e. "cm:content.size" and "cm:content.mimetype")
|
|
* @param sortAsc Given a sort property, true => ascending, false => descending
|
|
* @param queryExecutionId If paging then can pass back the previous query execution (as a hint for possible query optimisation)
|
|
*
|
|
* @return Returns ScriptPagingNodes which includes a JavaScript array of child file/folder nodes for this nodes.
|
|
* Automatically retrieves all sub-types of cm:content and cm:folder, also removes
|
|
* system folder types from the results.
|
|
* This is equivalent to @see FileFolderService.listFiles() and @see FileFolderService.listFolders()
|
|
*
|
|
* @author janv
|
|
* @since 4.0
|
|
*/
|
|
public ScriptPagingNodes childFileFolders(boolean files, boolean folders, Object ignoreTypes, int skipOffset, int maxItems, int requestTotalCountMax, String sortProp, Boolean sortAsc, String queryExecutionId)
|
|
{
|
|
Object[] results;
|
|
|
|
Set<QName> ignoreTypeQNames = new HashSet<QName>(5);
|
|
|
|
// Add user defined types to ignore
|
|
if (ignoreTypes instanceof ScriptableObject)
|
|
{
|
|
Serializable types = getValueConverter().convertValueForRepo((ScriptableObject)ignoreTypes);
|
|
if (types instanceof List)
|
|
{
|
|
for (Serializable typeObj : (List<Serializable>)types)
|
|
{
|
|
ignoreTypeQNames.add(createQName(typeObj.toString()));
|
|
}
|
|
}
|
|
else if (types instanceof String)
|
|
{
|
|
ignoreTypeQNames.add(createQName(types.toString()));
|
|
}
|
|
}
|
|
else if (ignoreTypes instanceof String)
|
|
{
|
|
ignoreTypeQNames.add(createQName(ignoreTypes.toString()));
|
|
}
|
|
|
|
List<Pair<QName, Boolean>> sortProps = null; // note: null sortProps => get all in default sort order
|
|
if (sortProp != null)
|
|
{
|
|
sortProps = new ArrayList<Pair<QName, Boolean>>(1);
|
|
sortProps.add(new Pair<QName, Boolean>(createQName(sortProp), sortAsc));
|
|
}
|
|
|
|
PagingRequest pageRequest = new PagingRequest(skipOffset, maxItems, queryExecutionId);
|
|
pageRequest.setRequestTotalCountMax(requestTotalCountMax);
|
|
|
|
PagingResults<FileInfo> pageOfNodeInfos = null;
|
|
FileFilterMode.setClient(Client.script);
|
|
try
|
|
{
|
|
pageOfNodeInfos = this.fileFolderService.list(this.nodeRef, files, folders, null, ignoreTypeQNames, sortProps, pageRequest);
|
|
}
|
|
finally
|
|
{
|
|
FileFilterMode.clearClient();
|
|
}
|
|
|
|
List<FileInfo> nodeInfos = pageOfNodeInfos.getPage();
|
|
int size = nodeInfos.size();
|
|
results = new Object[size];
|
|
for (int i=0; i<size; i++)
|
|
{
|
|
FileInfo nodeInfo = nodeInfos.get(i);
|
|
results[i] = newInstance(nodeInfo, this.services, this.scope);
|
|
}
|
|
|
|
int totalResultCountLower = -1;
|
|
int totalResultCountUpper = -1;
|
|
|
|
Pair<Integer, Integer> totalResultCount = pageOfNodeInfos.getTotalResultCount();
|
|
if (totalResultCount != null)
|
|
{
|
|
totalResultCountLower = (totalResultCount.getFirst() != null ? totalResultCount.getFirst() : -1);
|
|
totalResultCountUpper = (totalResultCount.getSecond() != null ? totalResultCount.getSecond() : -1);
|
|
}
|
|
|
|
return new ScriptPagingNodes(Context.getCurrentContext().newArray(this.scope, results), pageOfNodeInfos.hasMoreItems(), totalResultCountLower, totalResultCountUpper);
|
|
}
|
|
|
|
/**
|
|
* Return the target associations from this Node. As a Map of assoc type to a JavaScript array of Nodes.
|
|
* The Map returned implements the Scriptable interface to allow access to the assoc arrays via JavaScript
|
|
* associative array access. This means associations of this node can be access thus:
|
|
* <code>node.assocs["translations"][0]</code>
|
|
*
|
|
* @return target associations as a Map of assoc name to a JavaScript array of Nodes.
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public Map<String, Object> getAssocs()
|
|
{
|
|
if (this.targetAssocs == null)
|
|
{
|
|
// this Map implements the Scriptable interface for native JS syntax property access
|
|
this.targetAssocs = new ScriptableQNameMap<String, Object>(this);
|
|
|
|
// get the list of target nodes for each association type
|
|
List<AssociationRef> refs = this.nodeService.getTargetAssocs(this.nodeRef, RegexQNamePattern.MATCH_ALL);
|
|
for (AssociationRef ref : refs)
|
|
{
|
|
String qname = ref.getTypeQName().toString();
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.targetAssocs.get(qname);
|
|
if (nodes == null)
|
|
{
|
|
// first access of the list for this qname
|
|
nodes = new ArrayList<ScriptNode>(4);
|
|
}
|
|
this.targetAssocs.put(ref.getTypeQName().toString(), nodes);
|
|
nodes.add(newInstance(ref.getTargetRef(), this.services, this.scope));
|
|
}
|
|
|
|
// convert each Node list into a JavaScript array object
|
|
for (String qname : this.targetAssocs.keySet())
|
|
{
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.targetAssocs.get(qname);
|
|
Object[] objs = nodes.toArray(new Object[nodes.size()]);
|
|
this.targetAssocs.put(qname, Context.getCurrentContext().newArray(this.scope, objs));
|
|
}
|
|
}
|
|
|
|
return this.targetAssocs;
|
|
}
|
|
|
|
public Map<String, Object> getAssociations()
|
|
{
|
|
return getAssocs();
|
|
}
|
|
|
|
/**
|
|
* Return the source associations to this Node. As a Map of assoc name to a JavaScript array of Nodes.
|
|
* The Map returned implements the Scriptable interface to allow access to the assoc arrays via JavaScript
|
|
* associative array access. This means source associations to this node can be access thus:
|
|
* <code>node.sourceAssocs["translations"][0]</code>
|
|
*
|
|
* @return source associations as a Map of assoc name to a JavaScript array of Nodes.
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public Map<String, Object> getSourceAssocs()
|
|
{
|
|
if (this.sourceAssocs == null)
|
|
{
|
|
// this Map implements the Scriptable interface for native JS syntax property access
|
|
this.sourceAssocs = new ScriptableQNameMap<String, Object>(this);
|
|
|
|
// get the list of source nodes for each association type
|
|
List<AssociationRef> refs = this.nodeService.getSourceAssocs(this.nodeRef, RegexQNamePattern.MATCH_ALL);
|
|
for (AssociationRef ref : refs)
|
|
{
|
|
String qname = ref.getTypeQName().toString();
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.sourceAssocs.get(qname);
|
|
if (nodes == null)
|
|
{
|
|
// first access of the list for this qname
|
|
nodes = new ArrayList<ScriptNode>(4);
|
|
this.sourceAssocs.put(ref.getTypeQName().toString(), nodes);
|
|
}
|
|
nodes.add(newInstance(ref.getSourceRef(), this.services, this.scope));
|
|
}
|
|
|
|
// convert each Node list into a JavaScript array object
|
|
for (String qname : this.sourceAssocs.keySet())
|
|
{
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.sourceAssocs.get(qname);
|
|
Object[] objs = nodes.toArray(new Object[nodes.size()]);
|
|
this.sourceAssocs.put(qname, Context.getCurrentContext().newArray(this.scope, objs));
|
|
}
|
|
}
|
|
|
|
return this.sourceAssocs;
|
|
}
|
|
|
|
public Map<String, Object> getSourceAssociations()
|
|
{
|
|
return getSourceAssocs();
|
|
}
|
|
|
|
/**
|
|
* Return the child associations from this Node. As a Map of assoc name to a JavaScript array of Nodes.
|
|
* The Map returned implements the Scriptable interface to allow access to the assoc arrays via JavaScript
|
|
* associative array access. This means associations of this node can be access thus:
|
|
* <code>node.childAssocs["contains"][0]</code>
|
|
*
|
|
* @return child associations as a Map of assoc name to a JavaScript array of Nodes.
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public Map<String, Object> getChildAssocs()
|
|
{
|
|
if (this.childAssocs == null)
|
|
{
|
|
// this Map implements the Scriptable interface for native JS syntax property access
|
|
this.childAssocs = new ScriptableQNameMap<String, Object>(this);
|
|
|
|
// get the list of child assoc nodes for each association type
|
|
List<ChildAssociationRef> refs = this.nodeService.getChildAssocs(nodeRef);
|
|
for (ChildAssociationRef ref : refs)
|
|
{
|
|
String qname = ref.getTypeQName().toString();
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.childAssocs.get(qname);
|
|
if (nodes == null)
|
|
{
|
|
// first access of the list for this qname
|
|
nodes = new ArrayList<ScriptNode>(4);
|
|
this.childAssocs.put(ref.getTypeQName().toString(), nodes);
|
|
}
|
|
nodes.add(newInstance(ref.getChildRef(), this.services, this.scope));
|
|
}
|
|
|
|
// convert each Node list into a JavaScript array object
|
|
for (String qname : this.childAssocs.keySet())
|
|
{
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.childAssocs.get(qname);
|
|
Object[] objs = nodes.toArray(new Object[nodes.size()]);
|
|
this.childAssocs.put(qname, Context.getCurrentContext().newArray(this.scope, objs));
|
|
}
|
|
}
|
|
|
|
return this.childAssocs;
|
|
}
|
|
|
|
public Map<String, Object> getChildAssociations()
|
|
{
|
|
return getChildAssocs();
|
|
}
|
|
|
|
/**
|
|
* Return an Array of the associations from this Node that match a specific object type.
|
|
* <code>node.getChildAssocsByType("cm:folder")[0]</code>
|
|
*
|
|
* @return Array of child associations from this Node that match a specific object type.
|
|
*/
|
|
public Scriptable getChildAssocsByType(String type)
|
|
{
|
|
// get the list of child assoc nodes for each association type
|
|
Set<QName> types = new HashSet<QName>(1, 1.0f);
|
|
types.add(createQName(type));
|
|
List<ChildAssociationRef> refs = this.nodeService.getChildAssocs(this.nodeRef, types);
|
|
Object[] nodes = new Object[refs.size()];
|
|
for (int i=0; i<nodes.length; i++)
|
|
{
|
|
ChildAssociationRef ref = refs.get(i);
|
|
nodes[i] = newInstance(ref.getChildRef(), this.services, this.scope);
|
|
}
|
|
return Context.getCurrentContext().newArray(this.scope, nodes);
|
|
}
|
|
|
|
/**
|
|
* Return the parent associations to this Node. As a Map of assoc name to a JavaScript array of Nodes.
|
|
* The Map returned implements the Scriptable interface to allow access to the assoc arrays via JavaScript
|
|
* associative array access. This means associations of this node can be access thus:
|
|
* <code>node.parentAssocs["contains"][0]</code>
|
|
*
|
|
* @return parent associations as a Map of assoc name to a JavaScript array of Nodes.
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public Map<String, Object> getParentAssocs()
|
|
{
|
|
if (this.parentAssocs == null)
|
|
{
|
|
// this Map implements the Scriptable interface for native JS syntax property access
|
|
this.parentAssocs = new ScriptableQNameMap<String, Object>(this);
|
|
|
|
// get the list of child assoc nodes for each association type
|
|
List<ChildAssociationRef> refs = this.nodeService.getParentAssocs(nodeRef);
|
|
for (ChildAssociationRef ref : refs)
|
|
{
|
|
String qname = ref.getTypeQName().toString();
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.parentAssocs.get(qname);
|
|
if (nodes == null)
|
|
{
|
|
// first access of the list for this qname
|
|
nodes = new ArrayList<ScriptNode>(4);
|
|
this.parentAssocs.put(ref.getTypeQName().toString(), nodes);
|
|
}
|
|
nodes.add(newInstance(ref.getParentRef(), this.services, this.scope));
|
|
}
|
|
|
|
// convert each Node list into a JavaScript array object
|
|
for (String qname : this.parentAssocs.keySet())
|
|
{
|
|
List<ScriptNode> nodes = (List<ScriptNode>)this.parentAssocs.get(qname);
|
|
Object[] objs = nodes.toArray(new Object[nodes.size()]);
|
|
this.parentAssocs.put(qname, Context.getCurrentContext().newArray(this.scope, objs));
|
|
}
|
|
}
|
|
|
|
return this.parentAssocs;
|
|
}
|
|
|
|
public Map<String, Object> getParentAssociations()
|
|
{
|
|
return getParentAssocs();
|
|
}
|
|
|
|
/**
|
|
* Checks whether the {@link ScriptNode} exists in the repository.
|
|
* @return
|
|
*/
|
|
public boolean exists()
|
|
{
|
|
return nodeService.exists(nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Return all the properties known about this node. The Map returned implements the Scriptable interface to
|
|
* allow access to the properties via JavaScript associative array access. This means properties of a node can
|
|
* be access thus: <code>node.properties["name"]</code>
|
|
*
|
|
* @return Map of properties for this Node.
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public Map<String, Object> getProperties()
|
|
{
|
|
if (this.properties == null)
|
|
{
|
|
// this Map implements the Scriptable interface for native JS syntax property access
|
|
// this impl of the QNameMap is capable of creating ScriptContentData on demand for 'cm:content'
|
|
// properties that have not been initialised - see AR-1673.
|
|
this.properties = new ContentAwareScriptableQNameMap<String, Serializable>(this, this.services);
|
|
|
|
Map<QName, Serializable> props = null;
|
|
if (nodeInfo != null)
|
|
{
|
|
props = nodeInfo.getProperties();
|
|
}
|
|
else
|
|
{
|
|
props = this.nodeService.getProperties(this.nodeRef);
|
|
}
|
|
|
|
for (QName qname : props.keySet())
|
|
{
|
|
Serializable propValue = props.get(qname);
|
|
|
|
// perform the conversion to a script safe value and store
|
|
this.properties.put(qname.toString(), getValueConverter().convertValueForScript(qname, propValue));
|
|
}
|
|
}
|
|
|
|
return this.properties;
|
|
}
|
|
|
|
/**
|
|
* Return all the property names defined for this node's type as an array of short QNames.
|
|
*
|
|
* @return Array of property names for this node's type.
|
|
*/
|
|
public Scriptable getTypePropertyNames()
|
|
{
|
|
return getTypePropertyNames(true);
|
|
}
|
|
|
|
/**
|
|
* Return all the property names defined for this node's type as an array.
|
|
*
|
|
* @param useShortQNames if true short-form qnames will be returned, else long-form.
|
|
* @return Array of property names for this node's type.
|
|
*/
|
|
public Scriptable getTypePropertyNames(boolean useShortQNames)
|
|
{
|
|
Set<QName> props = this.services.getDictionaryService().getClass(this.getQNameType()).getProperties().keySet();
|
|
Object[] result = new Object[props.size()];
|
|
int count = 0;
|
|
for (QName qname : props)
|
|
{
|
|
result[count++] = useShortQNames ? getShortQName(qname).toString() : qname.toString();
|
|
}
|
|
return Context.getCurrentContext().newArray(this.scope, result);
|
|
}
|
|
|
|
/**
|
|
* Return all the property names defined for this node as an array.
|
|
*
|
|
* @param useShortQNames if true short-form qnames will be returned, else long-form.
|
|
* @return Array of property names for this node type and optionally parent properties.
|
|
*/
|
|
public Scriptable getPropertyNames(boolean useShortQNames)
|
|
{
|
|
Set<QName> props = this.nodeService.getProperties(this.nodeRef).keySet();
|
|
Object[] result = new Object[props.size()];
|
|
int count = 0;
|
|
for (QName qname : props)
|
|
{
|
|
result[count++] = useShortQNames ? getShortQName(qname).toString() : qname.toString();
|
|
}
|
|
return Context.getCurrentContext().newArray(this.scope, result);
|
|
}
|
|
|
|
/**
|
|
* @return true if this Node is a container (i.e. a folder)
|
|
*/
|
|
public boolean getIsContainer()
|
|
{
|
|
if (isContainer == null)
|
|
{
|
|
DictionaryService dd = this.services.getDictionaryService();
|
|
isContainer = Boolean.valueOf((dd.isSubClass(getQNameType(), ContentModel.TYPE_FOLDER) == true &&
|
|
dd.isSubClass(getQNameType(), ContentModel.TYPE_SYSTEM_FOLDER) == false));
|
|
}
|
|
|
|
return isContainer.booleanValue();
|
|
}
|
|
|
|
/**
|
|
* @return true if this Node is a Document (i.e. with content)
|
|
*/
|
|
public boolean getIsDocument()
|
|
{
|
|
if (isDocument == null)
|
|
{
|
|
DictionaryService dd = this.services.getDictionaryService();
|
|
isDocument = Boolean.valueOf(dd.isSubClass(getQNameType(), ContentModel.TYPE_CONTENT));
|
|
}
|
|
|
|
return isDocument.booleanValue();
|
|
}
|
|
|
|
/**
|
|
* @return true if this Node is a Link to a Container (i.e. a folderlink)
|
|
*/
|
|
public boolean getIsLinkToContainer()
|
|
{
|
|
if (isLinkToContainer == null)
|
|
{
|
|
DictionaryService dd = this.services.getDictionaryService();
|
|
isLinkToContainer = Boolean.valueOf(dd.isSubClass(getQNameType(), ApplicationModel.TYPE_FOLDERLINK));
|
|
}
|
|
|
|
return isLinkToContainer.booleanValue();
|
|
}
|
|
|
|
/**
|
|
* @return true if this Node is a Link to a Document (i.e. a filelink)
|
|
*/
|
|
public boolean getIsLinkToDocument()
|
|
{
|
|
if (isLinkToDocument == null)
|
|
{
|
|
DictionaryService dd = this.services.getDictionaryService();
|
|
isLinkToDocument = Boolean.valueOf(dd.isSubClass(getQNameType(), ApplicationModel.TYPE_FILELINK));
|
|
}
|
|
|
|
return isLinkToDocument.booleanValue();
|
|
}
|
|
|
|
/**
|
|
* @return true if the Node is a Category
|
|
*/
|
|
public boolean getIsCategory()
|
|
{
|
|
// this valid is overriden by the CategoryNode sub-class
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @return The list of aspects applied to this node
|
|
*/
|
|
public Set<QName> getAspectsSet()
|
|
{
|
|
if (this.aspects == null)
|
|
{
|
|
this.aspects = this.nodeService.getAspects(this.nodeRef);
|
|
}
|
|
|
|
return this.aspects;
|
|
}
|
|
|
|
/**
|
|
* @return The array of aspects applied to this node
|
|
*/
|
|
public Scriptable getAspects()
|
|
{
|
|
Set<QName> aspects = getAspectsSet();
|
|
Object[] result = new Object[aspects.size()];
|
|
int count = 0;
|
|
for (QName qname : aspects)
|
|
{
|
|
result[count++] = qname.toString();
|
|
}
|
|
return Context.getCurrentContext().newArray(this.scope, result);
|
|
}
|
|
|
|
/**
|
|
* @param aspect The aspect name to test for (fully qualified or short-name form)
|
|
* @return true if the node has the aspect false otherwise
|
|
*/
|
|
public boolean hasAspect(String aspect)
|
|
{
|
|
return getAspectsSet().contains(createQName(aspect));
|
|
}
|
|
|
|
/**
|
|
* @param type The qname type to test this object against (fully qualified or short-name form)
|
|
* @return true if this Node is a sub-type of the specified class (or itself of that class)
|
|
*/
|
|
public boolean isSubType(String type)
|
|
{
|
|
ParameterCheck.mandatoryString("Type", type);
|
|
|
|
QName qnameType = createQName(type);
|
|
|
|
return this.services.getDictionaryService().isSubClass(getQNameType(), qnameType);
|
|
}
|
|
|
|
/**
|
|
* @return QName path to this node. This can be used for Lucene PATH: style queries
|
|
*/
|
|
public String getQnamePath()
|
|
{
|
|
return this.services.getNodeService().getPath(getNodeRef()).toPrefixString(this.services.getNamespaceService());
|
|
}
|
|
|
|
/**
|
|
* @return Display path to this node
|
|
*/
|
|
public String getDisplayPath()
|
|
{
|
|
if (displayPath == null)
|
|
{
|
|
displayPath = this.nodeService.getPath(this.nodeRef).toDisplayPath(
|
|
this.nodeService, this.services.getPermissionService());
|
|
}
|
|
|
|
return displayPath;
|
|
}
|
|
|
|
/**
|
|
* @return the small icon image for this node
|
|
*/
|
|
public String getIcon16()
|
|
{
|
|
return "/images/filetypes/_default.gif";
|
|
}
|
|
|
|
/**
|
|
* @return the large icon image for this node
|
|
*/
|
|
public String getIcon32()
|
|
{
|
|
return "/images/filetypes32/_default.gif";
|
|
}
|
|
|
|
/**
|
|
* @return true if the node is currently locked
|
|
*/
|
|
public boolean getIsLocked()
|
|
{
|
|
boolean locked = false;
|
|
|
|
if (getAspectsSet().contains(ContentModel.ASPECT_LOCKABLE))
|
|
{
|
|
LockStatus lockStatus = this.services.getLockService().getLockStatus(this.nodeRef);
|
|
if (lockStatus == LockStatus.LOCKED || lockStatus == LockStatus.LOCK_OWNER)
|
|
{
|
|
locked = true;
|
|
}
|
|
}
|
|
|
|
return locked;
|
|
}
|
|
|
|
/**
|
|
* @return the primary parent node
|
|
*/
|
|
public ScriptNode getParent()
|
|
{
|
|
if (parent == null)
|
|
{
|
|
NodeRef parentRef = getPrimaryParentAssoc().getParentRef();
|
|
// handle root node (no parent!)
|
|
if (parentRef != null)
|
|
{
|
|
parent = newInstance(parentRef, this.services, this.scope);
|
|
}
|
|
}
|
|
|
|
return parent;
|
|
}
|
|
|
|
/**
|
|
* @return all parent nodes
|
|
*/
|
|
public Scriptable getParents()
|
|
{
|
|
List<ChildAssociationRef> parentRefs = this.nodeService.getParentAssocs(this.nodeRef);
|
|
Object[] parents = new Object[parentRefs.size()];
|
|
for (int i = 0; i < parentRefs.size(); i++)
|
|
{
|
|
NodeRef ref = parentRefs.get(i).getParentRef();
|
|
parents[i] = newInstance(ref, this.services, this.scope);
|
|
}
|
|
return Context.getCurrentContext().newArray(this.scope, parents);
|
|
}
|
|
|
|
/**
|
|
* @return the primary parent association so we can get at the association QName and the association type QName.
|
|
*/
|
|
public ChildAssociationRef getPrimaryParentAssoc()
|
|
{
|
|
if (primaryParentAssoc == null)
|
|
{
|
|
primaryParentAssoc = this.nodeService.getPrimaryParent(nodeRef);
|
|
}
|
|
return primaryParentAssoc;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Content API
|
|
|
|
/**
|
|
* @return the content String for this node from the default content property (@see ContentModel.PROP_CONTENT)
|
|
*/
|
|
public String getContent()
|
|
{
|
|
String content = "";
|
|
|
|
ScriptContentData contentData = (ScriptContentData)getProperties().get(ContentModel.PROP_CONTENT);
|
|
if (contentData != null)
|
|
{
|
|
content = contentData.getContent();
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Set the content for this node
|
|
*
|
|
* @param content Content string to set
|
|
*/
|
|
public void setContent(String content)
|
|
{
|
|
ScriptContentData contentData = (ScriptContentData)getProperties().get(ContentModel.PROP_CONTENT);
|
|
if (contentData != null)
|
|
{
|
|
contentData.setContent(content);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return For a content document, this method returns the URL to the content stream for the default content
|
|
* property (@see ContentModel.PROP_CONTENT)
|
|
* <p>
|
|
* For a container node, this method return the URL to browse to the folder in the web-client
|
|
*/
|
|
public String getUrl()
|
|
{
|
|
if (getIsDocument() == true)
|
|
{
|
|
return MessageFormat.format(CONTENT_DEFAULT_URL, new Object[] { nodeRef.getStoreRef().getProtocol(),
|
|
nodeRef.getStoreRef().getIdentifier(), nodeRef.getId(),
|
|
URLEncoder.encode(getName())});
|
|
}
|
|
else
|
|
{
|
|
return MessageFormat.format(FOLDER_BROWSE_URL, new Object[] { nodeRef.getStoreRef().getProtocol(),
|
|
nodeRef.getStoreRef().getIdentifier(), nodeRef.getId() });
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return For a content document, this method returns the download URL to the content for
|
|
* the default content property (@see ContentModel.PROP_CONTENT)
|
|
* <p>
|
|
* For a container node, this method returns an empty string
|
|
*/
|
|
public String getDownloadUrl()
|
|
{
|
|
if (getIsDocument() == true)
|
|
{
|
|
return MessageFormat.format(CONTENT_DOWNLOAD_URL, new Object[] {
|
|
nodeRef.getStoreRef().getProtocol(),
|
|
nodeRef.getStoreRef().getIdentifier(),
|
|
nodeRef.getId(),
|
|
URLEncoder.encode(getName()) });
|
|
}
|
|
else
|
|
{
|
|
return "";
|
|
}
|
|
}
|
|
|
|
public String jsGet_downloadUrl()
|
|
{
|
|
return getDownloadUrl();
|
|
}
|
|
|
|
/**
|
|
* @return The WebDav cm:name based path to the content for the default content property
|
|
* (@see ContentModel.PROP_CONTENT)
|
|
*/
|
|
public String getWebdavUrl()
|
|
{
|
|
String url = "";
|
|
try
|
|
{
|
|
if (getIsContainer() || getIsDocument())
|
|
{
|
|
List<FileInfo> paths = this.services.getFileFolderService().getNamePath(null, getNodeRef());
|
|
|
|
// build up the webdav url
|
|
StringBuilder path = new StringBuilder(128);
|
|
path.append("/webdav");
|
|
|
|
// build up the path skipping the first path as it is the root folder
|
|
for (int i=1; i<paths.size(); i++)
|
|
{
|
|
path.append("/")
|
|
.append(URLEncoder.encode(paths.get(i).getName()));
|
|
}
|
|
url = path.toString();
|
|
}
|
|
}
|
|
catch (InvalidTypeException typeErr)
|
|
{
|
|
// cannot build path if file is a type such as a rendition
|
|
}
|
|
catch (FileNotFoundException nodeErr)
|
|
{
|
|
// cannot build path if file no longer exists
|
|
}
|
|
return url;
|
|
}
|
|
|
|
/**
|
|
* @return The mimetype encoding for content attached to the node from the default content property
|
|
* (@see ContentModel.PROP_CONTENT)
|
|
*/
|
|
public String getMimetype()
|
|
{
|
|
String mimetype = null;
|
|
ScriptContentData content = (ScriptContentData) this.getProperties().get(ContentModel.PROP_CONTENT);
|
|
if (content != null)
|
|
{
|
|
mimetype = content.getMimetype();
|
|
}
|
|
|
|
return mimetype;
|
|
}
|
|
|
|
/**
|
|
* Set the mimetype encoding for the content attached to the node from the default content property
|
|
* (@see ContentModel.PROP_CONTENT)
|
|
*
|
|
* @param mimetype Mimetype to set
|
|
*/
|
|
public void setMimetype(String mimetype)
|
|
{
|
|
ScriptContentData content = (ScriptContentData) this.getProperties().get(ContentModel.PROP_CONTENT);
|
|
if (content != null)
|
|
{
|
|
content.setMimetype(mimetype);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return The size in bytes of the content attached to the node from the default content property
|
|
* (@see ContentModel.PROP_CONTENT)
|
|
*/
|
|
public long getSize()
|
|
{
|
|
long size = 0;
|
|
ScriptContentData content = (ScriptContentData) this.getProperties().get(ContentModel.PROP_CONTENT);
|
|
if (content != null)
|
|
{
|
|
size = content.getSize();
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Security API
|
|
|
|
/**
|
|
* Return true if the user has the specified permission on the node.
|
|
* <p>
|
|
* The default permissions are found in <code>org.alfresco.service.cmr.security.PermissionService</code>.
|
|
* Most commonly used are "Write", "Delete" and "AddChildren".
|
|
*
|
|
* @param permission as found in <code>org.alfresco.service.cmr.security.PermissionService</code>
|
|
* @return true if the user has the specified permission on the node.
|
|
*/
|
|
public boolean hasPermission(String permission)
|
|
{
|
|
ParameterCheck.mandatory("Permission Name", permission);
|
|
|
|
boolean allowed = false;
|
|
|
|
if (permission != null && permission.length() != 0)
|
|
{
|
|
AccessStatus status = this.services.getPermissionService().hasPermission(this.nodeRef, permission);
|
|
allowed = (AccessStatus.ALLOWED == status);
|
|
}
|
|
|
|
return allowed;
|
|
}
|
|
|
|
/**
|
|
* @return Array of permissions applied to this Node, including inherited.
|
|
* Strings returned are of the format [ALLOWED|DENIED];[USERNAME|GROUPNAME];PERMISSION for example
|
|
* ALLOWED;kevinr;Consumer so can be easily tokenized on the ';' character.
|
|
*/
|
|
public Scriptable getPermissions()
|
|
{
|
|
return Context.getCurrentContext().newArray(this.scope, retrieveAllSetPermissions(false, false));
|
|
}
|
|
|
|
/**
|
|
* @return Array of permissions applied directly to this Node (does not include inherited).
|
|
* Strings returned are of the format [ALLOWED|DENIED];[USERNAME|GROUPNAME];PERMISSION for example
|
|
* ALLOWED;kevinr;Consumer so can be easily tokenized on the ';' character.
|
|
*/
|
|
public Scriptable getDirectPermissions()
|
|
{
|
|
return Context.getCurrentContext().newArray(this.scope, retrieveAllSetPermissions(true, false));
|
|
}
|
|
|
|
/**
|
|
* @return Array of all permissions applied to this Node, including inherited.
|
|
* Strings returned are of the format [ALLOWED|DENIED];[USERNAME|GROUPNAME];PERMISSION;[INHERITED|DIRECT]
|
|
* for example: ALLOWED;kevinr;Consumer;DIRECT so can be easily tokenized on the ';' character.
|
|
*/
|
|
public Scriptable getFullPermissions()
|
|
{
|
|
return Context.getCurrentContext().newArray(this.scope, retrieveAllSetPermissions(false, true));
|
|
}
|
|
|
|
/**
|
|
* Helper to construct the response object for the various getPermissions() calls.
|
|
*
|
|
* @param direct True to only retrieve direct permissions, false to get inherited also
|
|
* @param full True to retrieve full data string with [INHERITED|DIRECT] element
|
|
* This exists to maintain backward compatibility with existing permission APIs.
|
|
*
|
|
* @return Object[] of packed permission strings.
|
|
*/
|
|
protected Object[] retrieveAllSetPermissions(boolean direct, boolean full)
|
|
{
|
|
Set<AccessPermission> acls = this.services.getPermissionService().getAllSetPermissions(getNodeRef());
|
|
List<Object> permissions = new ArrayList<Object>(acls.size());
|
|
for (AccessPermission permission : acls)
|
|
{
|
|
if (!direct || permission.isSetDirectly())
|
|
{
|
|
StringBuilder buf = new StringBuilder(64);
|
|
buf.append(permission.getAccessStatus())
|
|
.append(';')
|
|
.append(permission.getAuthority())
|
|
.append(';')
|
|
.append(permission.getPermission());
|
|
if (full)
|
|
{
|
|
buf.append(';').append(permission.isSetDirectly() ? "DIRECT" : "INHERITED");
|
|
}
|
|
permissions.add(buf.toString());
|
|
}
|
|
}
|
|
return (Object[])permissions.toArray(new Object[permissions.size()]);
|
|
}
|
|
|
|
/**
|
|
* @return Array of settable permissions for this Node
|
|
*/
|
|
public Scriptable getSettablePermissions()
|
|
{
|
|
Set<String> permissions = this.services.getPermissionService().getSettablePermissions(getNodeRef());
|
|
Object[] result = permissions.toArray(new Object[0]);
|
|
return Context.getCurrentContext().newArray(this.scope, result);
|
|
}
|
|
|
|
/**
|
|
* @return true if the node inherits permissions from the parent node, false otherwise
|
|
*/
|
|
public boolean inheritsPermissions()
|
|
{
|
|
return this.services.getPermissionService().getInheritParentPermissions(this.nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Set whether this node should inherit permissions from the parent node.
|
|
*
|
|
* @param inherit True to inherit parent permissions, false otherwise.
|
|
*/
|
|
public void setInheritsPermissions(boolean inherit)
|
|
{
|
|
this.services.getPermissionService().setInheritParentPermissions(this.nodeRef, inherit);
|
|
}
|
|
|
|
/**
|
|
* Apply a permission for ALL users to the node.
|
|
*
|
|
* @param permission Permission to apply
|
|
* @see org.alfresco.service.cmr.security.PermissionService
|
|
*/
|
|
public void setPermission(String permission)
|
|
{
|
|
ParameterCheck.mandatoryString("Permission Name", permission);
|
|
this.services.getPermissionService().setPermission(
|
|
this.nodeRef, PermissionService.ALL_AUTHORITIES, permission, true);
|
|
}
|
|
|
|
/**
|
|
* Apply a permission for the specified authority (e.g. username or group) to the node.
|
|
*
|
|
* @param permission Permission to apply @see org.alfresco.service.cmr.security.PermissionService
|
|
* @param authority Authority (generally a username or group name) to apply the permission for
|
|
*/
|
|
public void setPermission(String permission, String authority)
|
|
{
|
|
ParameterCheck.mandatoryString("Permission Name", permission);
|
|
ParameterCheck.mandatoryString("Authority", authority);
|
|
this.services.getPermissionService().setPermission(
|
|
this.nodeRef, authority, permission, true);
|
|
}
|
|
|
|
/**
|
|
* Remove a permission for ALL user from the node.
|
|
*
|
|
* @param permission Permission to remove @see org.alfresco.service.cmr.security.PermissionService
|
|
*/
|
|
public void removePermission(String permission)
|
|
{
|
|
ParameterCheck.mandatoryString("Permission Name", permission);
|
|
this.services.getPermissionService().deletePermission(
|
|
this.nodeRef, PermissionService.ALL_AUTHORITIES, permission);
|
|
}
|
|
|
|
/**
|
|
* Remove a permission for the specified authority (e.g. username or group) from the node.
|
|
*
|
|
* @param permission Permission to remove @see org.alfresco.service.cmr.security.PermissionService
|
|
* @param authority Authority (generally a username or group name) to apply the permission for
|
|
*/
|
|
public void removePermission(String permission, String authority)
|
|
{
|
|
ParameterCheck.mandatoryString("Permission Name", permission);
|
|
ParameterCheck.mandatoryString("Authority", authority);
|
|
this.services.getPermissionService().deletePermission(
|
|
this.nodeRef, authority, permission);
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Ownership API
|
|
|
|
/**
|
|
* Set the owner of the node
|
|
*/
|
|
public void setOwner(String userId)
|
|
{
|
|
this.services.getOwnableService().setOwner(this.nodeRef, userId);
|
|
}
|
|
|
|
/**
|
|
* Take ownership of the node.
|
|
*/
|
|
public void takeOwnership()
|
|
{
|
|
this.services.getOwnableService().takeOwnership(this.nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Get the owner of the node.
|
|
*
|
|
* @return
|
|
*/
|
|
public String getOwner()
|
|
{
|
|
return this.services.getOwnableService().getOwner(this.nodeRef);
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Create and Modify API
|
|
|
|
/**
|
|
* Persist the modified properties of this Node.
|
|
*/
|
|
public void save()
|
|
{
|
|
// persist properties back to the node in the DB
|
|
Map<QName, Serializable> props = new HashMap<QName, Serializable>(getProperties().size());
|
|
for (String key : this.properties.keySet())
|
|
{
|
|
Serializable value = (Serializable) this.properties.get(key);
|
|
|
|
// perform the conversion from script wrapper object to repo serializable values
|
|
value = getValueConverter().convertValueForRepo(value);
|
|
|
|
props.put(createQName(key), value);
|
|
}
|
|
this.nodeService.setProperties(this.nodeRef, props);
|
|
}
|
|
|
|
/**
|
|
* Re-sets the type of the node. Can be called in order specialise a node to a sub-type. This should be used
|
|
* with caution since calling it changes the type of the node and thus* implies a different set of aspects,
|
|
* properties and associations. It is the responsibility of the caller to ensure that the node is in a
|
|
* approriate state after changing the type.
|
|
*
|
|
* @param type Type to specialize the node
|
|
*
|
|
* @return true if successful, false otherwise
|
|
*/
|
|
public boolean specializeType(String type)
|
|
{
|
|
ParameterCheck.mandatoryString("Type", type);
|
|
|
|
QName qnameType = createQName(type);
|
|
|
|
// Ensure that we are performing a specialise
|
|
if (getQNameType().equals(qnameType) == false &&
|
|
this.services.getDictionaryService().isSubClass(qnameType, getQNameType()) == true)
|
|
{
|
|
// Specialise the type of the node
|
|
this.nodeService.setType(this.nodeRef, qnameType);
|
|
this.type = qnameType;
|
|
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Create a new File (cm:content) node as a child of this node.
|
|
* <p>
|
|
* Once created the file should have content set using the <code>content</code> property.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the file to create
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createFile(String name)
|
|
{
|
|
return createFile(name, null);
|
|
}
|
|
|
|
/**
|
|
* Create a new File (cm:content) node as a child of this node.
|
|
* <p>
|
|
* Once created the file should have content set using the <code>content</code> property.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the file to create
|
|
* @param type Type of the file to create (if null, defaults to ContentModel.TYPE_CONTENT)
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createFile(String name, String type)
|
|
{
|
|
ParameterCheck.mandatoryString("Node Name", name);
|
|
|
|
FileInfo fileInfo = this.services.getFileFolderService().create(
|
|
this.nodeRef, name, type == null ? ContentModel.TYPE_CONTENT : createQName(type));
|
|
|
|
reset();
|
|
|
|
ScriptNode file = newInstance(fileInfo.getNodeRef(), this.services, this.scope);
|
|
file.setMimetype(this.services.getMimetypeService().guessMimetype(name));
|
|
|
|
return file;
|
|
}
|
|
|
|
/**
|
|
* Create a new folder (cm:folder) node as a child of this node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the folder to create
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createFolder(String name)
|
|
{
|
|
return createFolder(name, null);
|
|
}
|
|
|
|
/**
|
|
* Create a new folder (cm:folder) node as a child of this node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the folder to create
|
|
* @param type Type of the folder to create (if null, defaults to ContentModel.TYPE_FOLDER)
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createFolder(String name, String type)
|
|
{
|
|
ParameterCheck.mandatoryString("Node Name", name);
|
|
|
|
FileInfo fileInfo = this.services.getFileFolderService().create(
|
|
this.nodeRef, name, type == null ? ContentModel.TYPE_FOLDER : createQName(type));
|
|
|
|
reset();
|
|
|
|
return newInstance(fileInfo.getNodeRef(), this.services, this.scope);
|
|
}
|
|
|
|
/**
|
|
* Create a new Node of the specified type as a child of this node.
|
|
*
|
|
* @param name Name of the node to create (can be null for a node without a 'cm:name' property)
|
|
* @param type QName type (fully qualified or short form such as 'cm:content')
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createNode(String name, String type)
|
|
{
|
|
return createNode(name, type, null, ContentModel.ASSOC_CONTAINS.toString());
|
|
}
|
|
|
|
/**
|
|
* Create a new Node of the specified type as a child of this node.
|
|
*
|
|
* @param name Name of the node to create (can be null for a node without a 'cm:name' property)
|
|
* @param type QName type (fully qualified or short form such as 'cm:content')
|
|
* @param assocType QName of the child association type (fully qualified or short form e.g. 'cm:contains')
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createNode(String name, String type, String assocType)
|
|
{
|
|
return createNode(name, type, null, assocType);
|
|
}
|
|
|
|
/**
|
|
* Create a new Node of the specified type as a child of this node.
|
|
*
|
|
* @param name Name of the node to create (can be null for a node without a 'cm:name' property)
|
|
* @param type QName type (fully qualified or short form such as 'cm:content')
|
|
* @param properties Associative array of the default properties for the node.
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createNode(String name, String type, Object properties)
|
|
{
|
|
return createNode(name, type, properties, ContentModel.ASSOC_CONTAINS.toString());
|
|
}
|
|
|
|
/**
|
|
* Create a new Node of the specified type as a child of this node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the node to create (can be null for a node without a 'cm:name' property)
|
|
* @param type QName type (fully qualified or short form such as 'cm:content')
|
|
* @param properties Associative array of the default properties for the node.
|
|
* @param assocType QName of the child association type (fully qualified or short form e.g. 'cm:contains')
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createNode(String name, String type, Object properties, String assocType)
|
|
{
|
|
return createNode(name, type, properties, assocType, null);
|
|
}
|
|
|
|
/**
|
|
* Create a new Node of the specified type as a child of this node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param name Name of the node to create (can be null for a node without a 'cm:name' property)
|
|
* @param type QName type (fully qualified or short form such as 'cm:content')
|
|
* @param properties Associative array of the default properties for the node.
|
|
* @param assocType QName of the child association type (fully qualified or short form e.g. 'cm:contains')
|
|
* @param assocName QName of the child association name (fully qualified or short form e.g. 'fm:discussion')
|
|
*
|
|
* @return Newly created Node or null if failed to create.
|
|
*/
|
|
public ScriptNode createNode(String name, String type, Object properties, String assocType, String assocName)
|
|
{
|
|
ParameterCheck.mandatoryString("Node Type", type);
|
|
ParameterCheck.mandatoryString("Association Type", assocType);
|
|
|
|
Map<QName, Serializable> props = null;
|
|
|
|
if (properties instanceof ScriptableObject)
|
|
{
|
|
props = new HashMap<QName, Serializable>(4, 1.0f);
|
|
extractScriptableProperties((ScriptableObject)properties, props);
|
|
}
|
|
|
|
if (name != null)
|
|
{
|
|
if (props == null) props = new HashMap<QName, Serializable>(1, 1.0f);
|
|
props.put(ContentModel.PROP_NAME, name);
|
|
}
|
|
else
|
|
{
|
|
// set name for the assoc local name
|
|
name = GUID.generate();
|
|
}
|
|
|
|
ChildAssociationRef childAssocRef = this.nodeService.createNode(
|
|
this.nodeRef,
|
|
createQName(assocType),
|
|
assocName == null ?
|
|
QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, QName.createValidLocalName(name)) :
|
|
createQName(assocName),
|
|
createQName(type),
|
|
props);
|
|
|
|
reset();
|
|
|
|
return newInstance(childAssocRef.getChildRef(), this.services, this.scope);
|
|
}
|
|
|
|
/**
|
|
* Creates a new secondary association between the current node and the specified child node.
|
|
* The association is given the same name as the child node's primary association.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param node node to add as a child of this node
|
|
*/
|
|
public void addNode(ScriptNode node)
|
|
{
|
|
ParameterCheck.mandatory("node", node);
|
|
ChildAssociationRef childAssocRef = this.nodeService.getPrimaryParent(node.nodeRef);
|
|
nodeService.addChild(this.nodeRef, node.nodeRef, ContentModel.ASSOC_CONTAINS, childAssocRef.getQName());
|
|
reset();
|
|
}
|
|
|
|
/**
|
|
* Remove an existing child node of this node.
|
|
*
|
|
* Severs all parent-child relationships between two nodes.
|
|
* <p>
|
|
* The child node will be cascade deleted if one of the associations was the
|
|
* primary association, i.e. the one with which the child node was created.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param node child node to remove
|
|
*/
|
|
public void removeNode(ScriptNode node)
|
|
{
|
|
ParameterCheck.mandatory("node", node);
|
|
nodeService.removeChild(this.nodeRef, node.nodeRef);
|
|
reset();
|
|
}
|
|
|
|
/**
|
|
* Create an association between this node and the specified target node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param target Destination node for the association
|
|
* @param assocType Association type qname (short form or fully qualified)
|
|
*/
|
|
public Association createAssociation(ScriptNode target, String assocType)
|
|
{
|
|
ParameterCheck.mandatory("Target", target);
|
|
ParameterCheck.mandatoryString("Association Type Name", assocType);
|
|
|
|
AssociationRef assocRef = this.nodeService.createAssociation(this.nodeRef, target.nodeRef, createQName(assocType));
|
|
reset();
|
|
return new Association(this.services, assocRef);
|
|
}
|
|
|
|
/**
|
|
* Remove an association between this node and the specified target node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param target Destination node on the end of the association
|
|
* @param assocType Association type qname (short form or fully qualified)
|
|
*/
|
|
public void removeAssociation(ScriptNode target, String assocType)
|
|
{
|
|
ParameterCheck.mandatory("Target", target);
|
|
ParameterCheck.mandatoryString("Association Type Name", assocType);
|
|
|
|
this.nodeService.removeAssociation(this.nodeRef, target.nodeRef, createQName(assocType));
|
|
reset();
|
|
}
|
|
|
|
/**
|
|
* Remove this node. Any references to this Node or its NodeRef should be discarded!
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
*/
|
|
public boolean remove()
|
|
{
|
|
boolean success = false;
|
|
|
|
if (nodeService.exists(this.nodeRef))
|
|
{
|
|
this.nodeService.deleteNode(this.nodeRef);
|
|
success = true;
|
|
}
|
|
|
|
reset();
|
|
|
|
return success;
|
|
}
|
|
|
|
/**
|
|
* Copy this Node to a new parent destination. Note that children of the source Node are not copied.
|
|
*
|
|
* @param destination Node
|
|
*
|
|
* @return The newly copied Node instance or null if failed to copy.
|
|
*/
|
|
public ScriptNode copy(ScriptNode destination)
|
|
{
|
|
ScriptNode copy = copy(destination, false);
|
|
|
|
// ALF-9517 fix
|
|
if (copy != null && copy.hasAspect(ContentModel.ASPECT_VERSIONABLE.toString()))
|
|
{
|
|
copy.ensureVersioningEnabled(true, true);
|
|
}
|
|
return copy;
|
|
}
|
|
|
|
/**
|
|
* Copy this Node and potentially all child nodes to a new parent destination.
|
|
*
|
|
* @param destination Node
|
|
* @param deepCopy True for a deep copy, false otherwise.
|
|
*
|
|
* @return The newly copied Node instance or null if failed to copy.
|
|
*/
|
|
public ScriptNode copy(ScriptNode destination, boolean deepCopy)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
|
|
ScriptNode copy = null;
|
|
|
|
if (destination.getNodeRef().getStoreRef().getProtocol().equals(StoreRef.PROTOCOL_WORKSPACE))
|
|
{
|
|
NodeRef copyRef = this.services.getCopyService().copyAndRename(this.nodeRef, destination.getNodeRef(),
|
|
ContentModel.ASSOC_CONTAINS, null, deepCopy);
|
|
copy = newInstance(copyRef, this.services, this.scope);
|
|
}
|
|
else
|
|
{
|
|
// NOTE: the deepCopy flag is not respected for this copy mechanism
|
|
copy = getCrossRepositoryCopyHelper().copy(this, destination, getName());
|
|
}
|
|
|
|
return copy;
|
|
}
|
|
|
|
/**
|
|
* Revert this Node to the specified version. Note this is not a deep revert of
|
|
* associations.
|
|
* This node must have the cm:versionable aspect. It will be checked out if required
|
|
* but will be checked in after the call.
|
|
*
|
|
* @param versionLabel to revert from
|
|
*
|
|
* @return the original Node that was checked out if reverted, {@code null} otherwise
|
|
* (if the version does not exist).
|
|
*/
|
|
public ScriptNode revert(String history, boolean majorVersion, String versionLabel)
|
|
{
|
|
return revert(history, majorVersion, versionLabel, false);
|
|
}
|
|
|
|
/**
|
|
* Revert this Node to the specified version and potentially all child nodes.
|
|
* This node must have the cm:versionable aspect. It will be checked out if required
|
|
* but will be checked in after the call.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param history Version history note
|
|
* @param majorVersion True to save as a major version increment, false for minor version.
|
|
* @param versionLabel to revert from
|
|
* @param deep {@code true} for a deep revert, {@code false} otherwise.
|
|
*
|
|
* @return the original Node that was checked out if reverted, {@code null} otherwise
|
|
* (if the version does not exist).
|
|
*/
|
|
public ScriptNode revert(String history, boolean majorVersion, String versionLabel, boolean deep)
|
|
{
|
|
if (!getIsVersioned())
|
|
{
|
|
return null;
|
|
}
|
|
|
|
// Get the Version - needed to do the revert
|
|
Version version = services.getVersionService().getVersionHistory(nodeRef).getVersion(versionLabel);
|
|
if (version == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
ScriptNode originalNode = this;
|
|
//cancel editing if we want to revert
|
|
if (nodeService.hasAspect(nodeRef, ContentModel.ASPECT_WORKING_COPY))
|
|
{
|
|
originalNode = cancelCheckout();
|
|
}
|
|
|
|
// Revert the new (current) version of the node
|
|
services.getVersionService().revert(originalNode.getNodeRef(), version, deep);
|
|
|
|
// Checkout/Checkin the node - to store the new version in version history
|
|
ScriptNode workingCopy = originalNode.checkout();
|
|
originalNode = workingCopy.checkin(history, majorVersion);
|
|
|
|
|
|
return originalNode;
|
|
}
|
|
|
|
/**
|
|
* Move this Node to a new parent destination.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param destination Node
|
|
*
|
|
* @return true on successful move, false on failure to move.
|
|
*/
|
|
public boolean move(ScriptNode destination)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
|
|
this.primaryParentAssoc = this.nodeService.moveNode(this.nodeRef, destination.getNodeRef(),
|
|
ContentModel.ASSOC_CONTAINS, getPrimaryParentAssoc().getQName());
|
|
|
|
// reset cached values
|
|
reset();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Move this Node from specified parent to a new parent destination.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param source Node
|
|
* @param destination Node
|
|
* @return true on successful move, false on failure to move.
|
|
*/
|
|
public boolean move(ScriptNode source, ScriptNode destination)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
|
|
if (source == null)
|
|
{
|
|
return move(destination);
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
this.services.getFileFolderService().moveFrom(this.nodeRef, source.getNodeRef(), destination.getNodeRef(), null);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
throw new ScriptException("Can't move node", e);
|
|
}
|
|
}
|
|
|
|
// reset cached values
|
|
reset();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Add an aspect to the Node. As no properties are provided in this call, it can only be used to add aspects that do not require any mandatory properties.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
* @param type Type name of the aspect to add
|
|
*
|
|
* @return true if the aspect was added successfully, false if an error occured.
|
|
*/
|
|
public boolean addAspect(String type)
|
|
{
|
|
return addAspect(type, null);
|
|
}
|
|
|
|
/**
|
|
* Add an aspect to the Node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param type Type name of the aspect to add
|
|
* @param props ScriptableObject (generally an assocative array) providing the named properties for the aspect
|
|
* - any mandatory properties for the aspect must be provided!
|
|
*
|
|
* @return true if the aspect was added successfully, false if an error occured.
|
|
*/
|
|
public boolean addAspect(String type, Object props)
|
|
{
|
|
ParameterCheck.mandatoryString("Aspect Type", type);
|
|
|
|
Map<QName, Serializable> aspectProps = null;
|
|
if (props instanceof ScriptableObject)
|
|
{
|
|
aspectProps = new HashMap<QName, Serializable>(4, 1.0f);
|
|
extractScriptableProperties((ScriptableObject)props, aspectProps);
|
|
}
|
|
QName aspectQName = createQName(type);
|
|
if (aspectQName.equals(ContentModel.ASPECT_VERSIONABLE))
|
|
{
|
|
// ALF-13719 need to taking into account script properties for versionable aspect
|
|
if (aspectProps != null)
|
|
{
|
|
Serializable autoVersionObj, autoVersionPropsObj;
|
|
autoVersionObj = aspectProps.get(ContentModel.PROP_AUTO_VERSION);
|
|
autoVersionPropsObj = aspectProps.get(ContentModel.PROP_AUTO_VERSION_PROPS);
|
|
ensureVersioningEnabled(autoVersionObj instanceof Boolean ? ((Boolean) autoVersionObj) : true,
|
|
autoVersionPropsObj instanceof Boolean ? ((Boolean) autoVersionPropsObj) : true);
|
|
}
|
|
else
|
|
{
|
|
ensureVersioningEnabled(true, true);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
this.nodeService.addAspect(this.nodeRef, aspectQName, aspectProps);
|
|
}
|
|
|
|
// reset the relevant cached node members
|
|
reset();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Extract a map of properties from a scriptable object (generally an associative array)
|
|
*
|
|
* @param scriptable The scriptable object to extract name/value pairs from.
|
|
* @param map The map to add the converted name/value pairs to.
|
|
*/
|
|
private void extractScriptableProperties(ScriptableObject scriptable, Map<QName, Serializable> map)
|
|
{
|
|
// we need to get all the keys to the properties provided
|
|
// and convert them to a Map of QName to Serializable objects
|
|
Object[] propIds = scriptable.getIds();
|
|
for (int i = 0; i < propIds.length; i++)
|
|
{
|
|
// work on each key in turn
|
|
Object propId = propIds[i];
|
|
|
|
// we are only interested in keys that are formed of Strings i.e. QName.toString()
|
|
if (propId instanceof String)
|
|
{
|
|
// get the value out for the specified key - it must be Serializable
|
|
String key = (String)propId;
|
|
Object value = scriptable.get(key, scriptable);
|
|
if (value instanceof Serializable)
|
|
{
|
|
value = getValueConverter().convertValueForRepo((Serializable)value);
|
|
map.put(createQName(key), (Serializable)value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove aspect from the node.
|
|
*
|
|
* Beware: Any unsaved property changes will be lost when this is called. To preserve property changes call {@link save()} first.
|
|
*
|
|
* @param type the aspect type
|
|
*
|
|
* @return true if successful, false otherwise
|
|
*/
|
|
public boolean removeAspect(String type)
|
|
{
|
|
ParameterCheck.mandatoryString("Aspect Type", type);
|
|
|
|
QName aspectQName = createQName(type);
|
|
this.nodeService.removeAspect(this.nodeRef, aspectQName);
|
|
|
|
// reset the relevant cached node members
|
|
reset();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Checkout/Checkin Services
|
|
|
|
/**
|
|
* Ensures that this document has the cm:versionable aspect applied to it,
|
|
* and that it has the initial version in the version store.
|
|
* Calling this on a versioned node with a version store entry will have
|
|
* no effect.
|
|
* Calling this on a newly uploaded share node will have versioning enabled
|
|
* for it (Share currently does lazy versioning to improve performance of
|
|
* documents that are uploaded but never edited, and multi upload performance).
|
|
*
|
|
* @param autoVersion If the cm:versionable aspect is applied, should auto versioning be requested?
|
|
* @param autoVersionProps If the cm:versionable aspect is applied, should auto versioning of properties be requested?
|
|
*/
|
|
public void ensureVersioningEnabled(boolean autoVersion, boolean autoVersionProps)
|
|
{
|
|
Map<QName, Serializable> props = new HashMap<QName, Serializable>(1, 1.0f);
|
|
props.put(ContentModel.PROP_AUTO_VERSION, autoVersion);
|
|
props.put(ContentModel.PROP_AUTO_VERSION_PROPS, autoVersionProps);
|
|
|
|
this.services.getVersionService().ensureVersioningEnabled(nodeRef, props);
|
|
}
|
|
|
|
/**
|
|
* Create a version of this document. Note: this will add the cm:versionable aspect.
|
|
*
|
|
* @param history Version history note
|
|
* @param majorVersion True to save as a major version increment, false for minor version.
|
|
*
|
|
* @return ScriptVersion object representing the newly added version node
|
|
*/
|
|
public ScriptVersion createVersion(String history, boolean majorVersion)
|
|
{
|
|
Map<String, Serializable> props = new HashMap<String, Serializable>(2, 1.0f);
|
|
props.put(Version.PROP_DESCRIPTION, history);
|
|
props.put(VersionModel.PROP_VERSION_TYPE, majorVersion ? VersionType.MAJOR : VersionType.MINOR);
|
|
ScriptVersion version = new ScriptVersion(this.services.getVersionService().createVersion(this.nodeRef, props), this.services, this.scope);
|
|
this.versions = null;
|
|
return version;
|
|
}
|
|
|
|
/**
|
|
* Determines if this node is versioned
|
|
*
|
|
* @return true => is versioned
|
|
*/
|
|
public boolean getIsVersioned()
|
|
{
|
|
return this.nodeService.hasAspect(this.nodeRef, ContentModel.ASPECT_VERSIONABLE);
|
|
}
|
|
|
|
/**
|
|
* Gets the version history
|
|
*
|
|
* @return version history
|
|
*/
|
|
public Scriptable getVersionHistory()
|
|
{
|
|
if (this.versions == null && getIsVersioned())
|
|
{
|
|
VersionHistory history = this.services.getVersionService().getVersionHistory(this.nodeRef);
|
|
if (history != null)
|
|
{
|
|
Collection<Version> allVersions = history.getAllVersions();
|
|
Object[] versions = new Object[allVersions.size()];
|
|
int i = 0;
|
|
for (Version version : allVersions)
|
|
{
|
|
versions[i++] = new ScriptVersion(version, this.services, this.scope);
|
|
}
|
|
this.versions = Context.getCurrentContext().newArray(this.scope, versions);
|
|
}
|
|
}
|
|
return this.versions;
|
|
}
|
|
|
|
/**
|
|
* Gets the version of this node specified by version label
|
|
*
|
|
* @param versionLabel version label
|
|
* @return version of node, or null if node is not versioned, or label does not exist
|
|
*/
|
|
public ScriptVersion getVersion(String versionLabel)
|
|
{
|
|
if (!getIsVersioned())
|
|
{
|
|
return null;
|
|
}
|
|
VersionHistory history = this.services.getVersionService().getVersionHistory(this.nodeRef);
|
|
Version version = history.getVersion(versionLabel);
|
|
if (version == null)
|
|
{
|
|
return null;
|
|
}
|
|
return new ScriptVersion(version, this.services, this.scope);
|
|
}
|
|
|
|
/**
|
|
* Perform a check-out of this document into the current parent space.
|
|
*
|
|
* @return the working copy Node for the checked out document
|
|
*/
|
|
public ScriptNode checkout()
|
|
{
|
|
NodeRef workingCopyRef = this.services.getCheckOutCheckInService().checkout(this.nodeRef);
|
|
ScriptNode workingCopy = newInstance(workingCopyRef, this.services, this.scope);
|
|
|
|
// reset the aspect and properties as checking out a document causes changes
|
|
this.properties = null;
|
|
this.aspects = null;
|
|
|
|
return workingCopy;
|
|
}
|
|
|
|
/**
|
|
* Performs a check-out of this document for the purposes of an upload
|
|
*
|
|
* @return
|
|
*/
|
|
public ScriptNode checkoutForUpload()
|
|
{
|
|
AlfrescoTransactionSupport.bindResource("checkoutforupload", Boolean.TRUE.toString());
|
|
services.getRuleService().disableRules();
|
|
try
|
|
{
|
|
return checkout();
|
|
}
|
|
finally
|
|
{
|
|
services.getRuleService().enableRules();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Perform a check-out of this document into the specified destination space.
|
|
*
|
|
* @param destination
|
|
* Destination for the checked out document working copy Node.
|
|
* @return the working copy Node for the checked out document
|
|
*/
|
|
public ScriptNode checkout(ScriptNode destination)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
|
|
ChildAssociationRef childAssocRef = this.nodeService.getPrimaryParent(destination.getNodeRef());
|
|
NodeRef workingCopyRef = this.services.getCheckOutCheckInService().checkout(this.nodeRef,
|
|
destination.getNodeRef(), ContentModel.ASSOC_CONTAINS, childAssocRef.getQName());
|
|
ScriptNode workingCopy = newInstance(workingCopyRef, this.services, this.scope);
|
|
|
|
// reset the aspect and properties as checking out a document causes changes
|
|
this.properties = null;
|
|
this.aspects = null;
|
|
|
|
return workingCopy;
|
|
}
|
|
|
|
/**
|
|
* Check-in a working copy document. The current state of the working copy is copied to the original node,
|
|
* this will include any content updated in the working node. Note that this method can only be called on a
|
|
* working copy Node.
|
|
*
|
|
* @return the original Node that was checked out.
|
|
*/
|
|
public ScriptNode checkin()
|
|
{
|
|
return checkin("", false);
|
|
}
|
|
|
|
/**
|
|
* Check-in a working copy document. The current state of the working copy is copied to the original node,
|
|
* this will include any content updated in the working node. Note that this method can only be called on a
|
|
* working copy Node.
|
|
*
|
|
* @param history Version history note
|
|
*
|
|
* @return the original Node that was checked out.
|
|
*/
|
|
public ScriptNode checkin(String history)
|
|
{
|
|
return checkin(history, false);
|
|
}
|
|
|
|
/**
|
|
* Check-in a working copy document. The current state of the working copy is copied to the original node,
|
|
* this will include any content updated in the working node. Note that this method can only be called on a
|
|
* working copy Node.
|
|
*
|
|
* @param history Version history note
|
|
* @param majorVersion True to save as a major version increment, false for minor version.
|
|
*
|
|
* @return the original Node that was checked out.
|
|
*/
|
|
public ScriptNode checkin(String history, boolean majorVersion)
|
|
{
|
|
Map<String, Serializable> props = new HashMap<String, Serializable>(2, 1.0f);
|
|
props.put(Version.PROP_DESCRIPTION, history);
|
|
props.put(VersionModel.PROP_VERSION_TYPE, majorVersion ? VersionType.MAJOR : VersionType.MINOR);
|
|
NodeRef original = this.services.getCheckOutCheckInService().checkin(this.nodeRef, props);
|
|
this.versions = null;
|
|
return newInstance(original, this.services, this.scope);
|
|
}
|
|
|
|
/**
|
|
* Removes the lock on a node.
|
|
*
|
|
*/
|
|
public void unlock()
|
|
{
|
|
this.services.getLockService().unlock(this.nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Cancel the check-out of a working copy document. The working copy will be deleted and any changes made to it
|
|
* are lost. Note that this method can only be called on a working copy Node. The reference to this working copy
|
|
* Node should be discarded.
|
|
*
|
|
* @return the original Node that was checked out.
|
|
*/
|
|
public ScriptNode cancelCheckout()
|
|
{
|
|
NodeRef original = this.services.getCheckOutCheckInService().cancelCheckout(this.nodeRef);
|
|
return newInstance(original, this.services, this.scope);
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Transformation and Rendering API
|
|
|
|
/**
|
|
* Transform a document to a new document mimetype format. A copy of the document is made and the extension
|
|
* changed to match the new mimetype, then the transformation isapplied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
*
|
|
* @return Node representing the newly transformed document.
|
|
*/
|
|
public ScriptNode transformDocument(String mimetype)
|
|
{
|
|
return transformDocument(mimetype, getPrimaryParentAssoc().getParentRef());
|
|
}
|
|
|
|
/**
|
|
* Transform a document to a new document mimetype format. A copy of the document is made in the specified
|
|
* destination folder and the extension changed to match the new mimetype, then then transformation is applied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
* @param destination Destination folder location
|
|
*
|
|
* @return Node representing the newly transformed document.
|
|
*/
|
|
public ScriptNode transformDocument(String mimetype, ScriptNode destination)
|
|
{
|
|
return transformDocument(mimetype, destination.getNodeRef());
|
|
}
|
|
|
|
private ScriptNode transformDocument(String mimetype, NodeRef destination)
|
|
{
|
|
ParameterCheck.mandatoryString("Mimetype", mimetype);
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
final NodeRef sourceNodeRef = nodeRef;
|
|
|
|
// the delegate definition for transforming a document
|
|
Transformer transformer = new Transformer()
|
|
{
|
|
public ScriptNode transform(ContentService contentService, NodeRef nodeRef, ContentReader reader,
|
|
ContentWriter writer)
|
|
{
|
|
ScriptNode transformedNode = null;
|
|
TransformationOptions options = new TransformationOptions();
|
|
options.setSourceNodeRef(sourceNodeRef);
|
|
|
|
try
|
|
{
|
|
contentService.transform(reader, writer, options);
|
|
transformedNode = newInstance(nodeRef, services, scope);
|
|
}
|
|
catch (NoTransformerException e)
|
|
{
|
|
// ignore
|
|
}
|
|
return transformedNode;
|
|
}
|
|
};
|
|
|
|
return transformNode(transformer, mimetype, destination);
|
|
}
|
|
|
|
/**
|
|
* Generic method to transform Node content from one mimetype to another.
|
|
*
|
|
* @param transformer The Transformer delegate supplying the transformation logic
|
|
* @param mimetype Mimetype of the destination content
|
|
* @param destination Destination folder location for the resulting document
|
|
*
|
|
* @return Node representing the transformed content - or null if the transform failed
|
|
*/
|
|
private ScriptNode transformNode(Transformer transformer, String mimetype, NodeRef destination)
|
|
{
|
|
ScriptNode transformedNode = null;
|
|
|
|
// get the content reader
|
|
ContentService contentService = this.services.getContentService();
|
|
ContentReader reader = contentService.getReader(this.nodeRef, ContentModel.PROP_CONTENT);
|
|
|
|
// only perform the transformation if some content is available
|
|
if (reader != null)
|
|
{
|
|
// Copy the content node to a new node
|
|
String copyName = TransformActionExecuter.transformName(this.services.getMimetypeService(), getName(),
|
|
mimetype, true);
|
|
NodeRef copyNodeRef = this.services.getCopyService().copy(this.nodeRef, destination,
|
|
ContentModel.ASSOC_CONTAINS,
|
|
QName.createQName(ContentModel.PROP_CONTENT.getNamespaceURI(), QName.createValidLocalName(copyName)),
|
|
false);
|
|
|
|
// modify the name of the copy to reflect the new mimetype
|
|
this.nodeService.setProperty(copyNodeRef, ContentModel.PROP_NAME, copyName);
|
|
|
|
// get the writer and set it up
|
|
ContentWriter writer = contentService.getWriter(copyNodeRef, ContentModel.PROP_CONTENT, true);
|
|
writer.setMimetype(mimetype); // new mimetype
|
|
writer.setEncoding(reader.getEncoding()); // original encoding
|
|
|
|
// Try and transform the content using the supplied delegate
|
|
transformedNode = transformer.transform(contentService, copyNodeRef, reader, writer);
|
|
}
|
|
|
|
return transformedNode;
|
|
}
|
|
|
|
/**
|
|
* Transform an image to a new image format. A copy of the image document is made and the extension changed to
|
|
* match the new mimetype, then the transformation is applied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
*
|
|
* @return Node representing the newly transformed image.
|
|
*/
|
|
public ScriptNode transformImage(String mimetype)
|
|
{
|
|
return transformImage(mimetype, null, getPrimaryParentAssoc().getParentRef());
|
|
}
|
|
|
|
/**
|
|
* Transform an image to a new image format. A copy of the image document is made and the extension changed to
|
|
* match the new mimetype, then the transformation is applied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
* @param options Image convert command options
|
|
*
|
|
* @return Node representing the newly transformed image.
|
|
*/
|
|
public ScriptNode transformImage(String mimetype, String options)
|
|
{
|
|
return transformImage(mimetype, options, getPrimaryParentAssoc().getParentRef());
|
|
}
|
|
|
|
/**
|
|
* Transform an image to a new image mimetype format. A copy of the image document is made in the specified
|
|
* destination folder and the extension changed to match the newmimetype, then then transformation is applied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
* @param destination Destination folder location
|
|
*
|
|
* @return Node representing the newly transformed image.
|
|
*/
|
|
public ScriptNode transformImage(String mimetype, ScriptNode destination)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
return transformImage(mimetype, null, destination.getNodeRef());
|
|
}
|
|
|
|
/**
|
|
* Transform an image to a new image mimetype format. A copy of the image document is made in the specified
|
|
* destination folder and the extension changed to match the new
|
|
* mimetype, then then transformation is applied.
|
|
*
|
|
* @param mimetype Mimetype destination for the transformation
|
|
* @param options Image convert command options
|
|
* @param destination Destination folder location
|
|
*
|
|
* @return Node representing the newly transformed image.
|
|
*/
|
|
public ScriptNode transformImage(String mimetype, String options, ScriptNode destination)
|
|
{
|
|
ParameterCheck.mandatory("Destination Node", destination);
|
|
return transformImage(mimetype, options, destination.getNodeRef());
|
|
}
|
|
|
|
private ScriptNode transformImage(String mimetype, final String options, NodeRef destination)
|
|
{
|
|
ParameterCheck.mandatoryString("Mimetype", mimetype);
|
|
final NodeRef sourceNodeRef = nodeRef;
|
|
|
|
// the delegate definition for transforming an image
|
|
Transformer transformer = new Transformer()
|
|
{
|
|
public ScriptNode transform(ContentService contentService, NodeRef nodeRef, ContentReader reader,
|
|
ContentWriter writer)
|
|
{
|
|
ImageTransformationOptions imageOptions = new ImageTransformationOptions();
|
|
imageOptions.setSourceNodeRef(sourceNodeRef);
|
|
|
|
if (options != null)
|
|
{
|
|
imageOptions.setCommandOptions(options);
|
|
}
|
|
contentService.getImageTransformer().transform(reader, writer, imageOptions);
|
|
|
|
return newInstance(nodeRef, services, scope);
|
|
}
|
|
};
|
|
|
|
return transformNode(transformer, mimetype, destination);
|
|
}
|
|
|
|
/**
|
|
* Process a FreeMarker Template against the current node.
|
|
*
|
|
* @param template Node of the template to execute
|
|
*
|
|
* @return output of the template execution
|
|
*/
|
|
public String processTemplate(ScriptNode template)
|
|
{
|
|
ParameterCheck.mandatory("Template Node", template);
|
|
return processTemplate(template.getContent(), null, null);
|
|
}
|
|
|
|
/**
|
|
* Process a FreeMarker Template against the current node.
|
|
*
|
|
* @param template Node of the template to execute
|
|
* @param args Scriptable object (generally an associative array) containing the name/value pairs of
|
|
* arguments to be passed to the template
|
|
*
|
|
* @return output of the template execution
|
|
*/
|
|
public String processTemplate(ScriptNode template, Object args)
|
|
{
|
|
ParameterCheck.mandatory("Template Node", template);
|
|
return processTemplate(template.getContent(), null, (ScriptableObject)args);
|
|
}
|
|
|
|
/**
|
|
* Process a FreeMarker Template against the current node.
|
|
*
|
|
* @param template The template to execute
|
|
*
|
|
* @return output of the template execution
|
|
*/
|
|
public String processTemplate(String template)
|
|
{
|
|
ParameterCheck.mandatoryString("Template", template);
|
|
return processTemplate(template, null, null);
|
|
}
|
|
|
|
/**
|
|
* Process a FreeMarker Template against the current node.
|
|
*
|
|
* @param template The template to execute
|
|
* @param args Scriptable object (generally an associative array) containing the name/value pairs of
|
|
* arguments to be passed to the template
|
|
*
|
|
* @return output of the template execution
|
|
*/
|
|
public String processTemplate(String template, Object args)
|
|
{
|
|
ParameterCheck.mandatoryString("Template", template);
|
|
return processTemplate(template, null, (ScriptableObject)args);
|
|
}
|
|
|
|
private String processTemplate(String template, NodeRef templateRef, ScriptableObject args)
|
|
{
|
|
Object person = (Object)scope.get("person", scope);
|
|
Object companyhome = (Object)scope.get("companyhome", scope);
|
|
Object userhome = (Object)scope.get("userhome", scope);
|
|
|
|
// build default model for the template processing
|
|
Map<String, Object> model = this.services.getTemplateService().buildDefaultModel(
|
|
(person.equals(UniqueTag.NOT_FOUND)) ? null : ((ScriptNode)((Wrapper)person).unwrap()).getNodeRef(),
|
|
(companyhome.equals(UniqueTag.NOT_FOUND)) ? null : ((ScriptNode)((Wrapper)companyhome).unwrap()).getNodeRef(),
|
|
(userhome.equals(UniqueTag.NOT_FOUND)) ? null : ((ScriptNode)((Wrapper)userhome).unwrap()).getNodeRef(),
|
|
templateRef,
|
|
null);
|
|
|
|
// add the current node as either the document/space as appropriate
|
|
if (this.getIsDocument())
|
|
{
|
|
model.put("document", this.nodeRef);
|
|
model.put("space", getPrimaryParentAssoc().getParentRef());
|
|
}
|
|
else
|
|
{
|
|
model.put("space", this.nodeRef);
|
|
}
|
|
|
|
// add the supplied args to the 'args' root object
|
|
if (args != null)
|
|
{
|
|
// we need to get all the keys to the properties provided
|
|
// and convert them to a Map of QName to Serializable objects
|
|
Object[] propIds = args.getIds();
|
|
Map<String, String> templateArgs = new HashMap<String, String>(propIds.length);
|
|
for (int i = 0; i < propIds.length; i++)
|
|
{
|
|
// work on each key in turn
|
|
Object propId = propIds[i];
|
|
|
|
// we are only interested in keys that are formed of Strings i.e. QName.toString()
|
|
if (propId instanceof String)
|
|
{
|
|
// get the value out for the specified key - make sure it is Serializable
|
|
Object value = args.get((String) propId, args);
|
|
value = getValueConverter().convertValueForRepo((Serializable)value);
|
|
if (value != null)
|
|
{
|
|
templateArgs.put((String) propId, value.toString());
|
|
}
|
|
}
|
|
}
|
|
// add the args to the model as the 'args' root object
|
|
model.put("args", templateArgs);
|
|
}
|
|
|
|
// execute template!
|
|
// TODO: check that script modified nodes are reflected...
|
|
return this.services.getTemplateService().processTemplateString(null, template, model);
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Thumbnail Methods
|
|
|
|
/**
|
|
* Creates a thumbnail for the content property of the node.
|
|
*
|
|
* The thumbnail name correspionds to pre-set thumbnail details stored in the
|
|
* repository.
|
|
*
|
|
* @param thumbnailName the name of the thumbnail
|
|
* @return ScriptThumbnail the newly create thumbnail node
|
|
*/
|
|
public ScriptThumbnail createThumbnail(String thumbnailName)
|
|
{
|
|
return createThumbnail(thumbnailName, false);
|
|
}
|
|
|
|
/**
|
|
* Creates a thumbnail for the content property of the node.
|
|
*
|
|
* The thumbnail name correspionds to pre-set thumbnail details stored in the
|
|
* repository.
|
|
*
|
|
* If the thumbnail is created asynchronously then the result will be null and creation
|
|
* of the thumbnail will occure at some point in the background.
|
|
*
|
|
* @param thumbnailName the name of the thumbnail
|
|
* @param async indicates whether the thumbnail is create asynchronously or not
|
|
* @return ScriptThumbnail the newly create thumbnail node or null if async creation occures
|
|
*/
|
|
public ScriptThumbnail createThumbnail(String thumbnailName, boolean async)
|
|
{
|
|
ScriptThumbnail result = null;
|
|
|
|
// Use the thumbnail registy to get the details of the thumbail
|
|
ThumbnailRegistry registry = this.services.getThumbnailService().getThumbnailRegistry();
|
|
ThumbnailDefinition details = registry.getThumbnailDefinition(thumbnailName);
|
|
if (details == null)
|
|
{
|
|
// Throw exception
|
|
throw new ScriptException("The thumbnail name '" + thumbnailName + "' is not registered");
|
|
}
|
|
|
|
// If there's nothing currently registered to generate thumbnails for the
|
|
// specified mimetype, then log a message and bail out
|
|
String nodeMimeType = getMimetype();
|
|
Serializable value = this.nodeService.getProperty(nodeRef, ContentModel.PROP_CONTENT);
|
|
ContentData contentData = DefaultTypeConverter.INSTANCE.convert(ContentData.class, value);
|
|
if (contentData == null)
|
|
{
|
|
logger.info("Unable to create thumbnail '" + details.getName() + "' as there is no content");
|
|
return null;
|
|
}
|
|
if (!registry.isThumbnailDefinitionAvailable(contentData.getContentUrl(), nodeMimeType, getSize(), nodeRef, details))
|
|
{
|
|
logger.info("Unable to create thumbnail '" + details.getName() + "' for " +
|
|
nodeMimeType + " as no transformer is currently available");
|
|
return null;
|
|
}
|
|
|
|
// Have the thumbnail created
|
|
if (async == false)
|
|
{
|
|
try
|
|
{
|
|
// Create the thumbnail
|
|
NodeRef thumbnailNodeRef = this.services.getThumbnailService().createThumbnail(
|
|
this.nodeRef,
|
|
ContentModel.PROP_CONTENT,
|
|
details.getMimetype(),
|
|
details.getTransformationOptions(),
|
|
details.getName());
|
|
|
|
// Create the thumbnail script object
|
|
result = new ScriptThumbnail(thumbnailNodeRef, this.services, this.scope);
|
|
}
|
|
catch (AlfrescoRuntimeException e)
|
|
{
|
|
Throwable rootCause = e.getRootCause();
|
|
if (rootCause instanceof UnimportantTransformException)
|
|
{
|
|
logger.debug("Unable to create thumbnail '" + details.getName() + "' as "+rootCause.getMessage());
|
|
return null;
|
|
}
|
|
throw e;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Action action = ThumbnailHelper.createCreateThumbnailAction(details, services);
|
|
|
|
// Queue async creation of thumbnail
|
|
this.services.getActionService().executeAction(action, this.nodeRef, true, true);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Get the given thumbnail for the content property
|
|
*
|
|
* @param thumbnailName the thumbnail name
|
|
* @return ScriptThumbnail the thumbnail
|
|
*/
|
|
public ScriptThumbnail getThumbnail(String thumbnailName)
|
|
{
|
|
ScriptThumbnail result = null;
|
|
NodeRef thumbnailNodeRef = this.services.getThumbnailService().getThumbnailByName(
|
|
this.nodeRef,
|
|
ContentModel.PROP_CONTENT,
|
|
thumbnailName);
|
|
if (thumbnailNodeRef != null)
|
|
{
|
|
result = new ScriptThumbnail(thumbnailNodeRef, this.services, this.scope);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Get the all the thumbnails for a given node's content property.
|
|
*
|
|
* @return Scriptable list of thumbnails, empty if none available
|
|
*/
|
|
public ScriptThumbnail[] getThumbnails()
|
|
{
|
|
List<NodeRef> thumbnails = this.services.getThumbnailService().getThumbnails(
|
|
this.nodeRef,
|
|
ContentModel.PROP_CONTENT,
|
|
null,
|
|
null);
|
|
|
|
List<ScriptThumbnail> result = new ArrayList<ScriptThumbnail>(thumbnails.size());
|
|
for (NodeRef thumbnail : thumbnails)
|
|
{
|
|
ScriptThumbnail scriptThumbnail = new ScriptThumbnail(thumbnail, this.services, this.scope);
|
|
result.add(scriptThumbnail);
|
|
}
|
|
return (ScriptThumbnail[])result.toArray(new ScriptThumbnail[result.size()]);
|
|
}
|
|
|
|
/**
|
|
* Returns the names of the thumbnail defintions that can be applied to the content property of
|
|
* this node.
|
|
* <p>
|
|
* Thumbanil defintions only appear in this list if they can produce a thumbnail for the content
|
|
* found in the content property. This will be determined by looking at the mimetype of the content
|
|
* and the destinatino mimetype of the thumbnail.
|
|
*
|
|
* @return String[] array of thumbnail names that are valid for the current content type
|
|
*/
|
|
public String[] getThumbnailDefinitions()
|
|
{
|
|
ContentService contentService = this.services.getContentService();
|
|
ThumbnailService thumbnailService = this.services.getThumbnailService();
|
|
|
|
List<String> result = new ArrayList<String>(7);
|
|
|
|
ContentReader contentReader = contentService.getReader(this.nodeRef, ContentModel.PROP_CONTENT);
|
|
if (contentReader != null)
|
|
{
|
|
String mimetype = contentReader.getMimetype();
|
|
List<ThumbnailDefinition> thumbnailDefinitions = thumbnailService.getThumbnailRegistry().getThumbnailDefinitions(mimetype, contentReader.getSize());
|
|
for (ThumbnailDefinition thumbnailDefinition : thumbnailDefinitions)
|
|
{
|
|
result.add(thumbnailDefinition.getName());
|
|
}
|
|
}
|
|
|
|
return (String[])result.toArray(new String[result.size()]);
|
|
}
|
|
/**
|
|
* This version of the method name spelling is retained (for now) for backwards compatibility
|
|
* @see #getThumbnailDefinitions()
|
|
*/
|
|
@Deprecated
|
|
public String[] getThumbnailDefintions()
|
|
{
|
|
return getThumbnailDefinitions();
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Tag methods
|
|
|
|
/**
|
|
* Clear the node's tags
|
|
*/
|
|
public void clearTags()
|
|
{
|
|
this.services.getTaggingService().clearTags(this.nodeRef);
|
|
updateTagProperty();
|
|
}
|
|
|
|
/**
|
|
* Adds a tag to the node
|
|
*
|
|
* @param tag tag name
|
|
*/
|
|
public void addTag(String tag)
|
|
{
|
|
this.services.getTaggingService().addTag(this.nodeRef, tag);
|
|
updateTagProperty();
|
|
}
|
|
|
|
/**
|
|
* Adds all the tags to the node
|
|
*
|
|
* @param tags array of tag names
|
|
*/
|
|
public void addTags(String[] tags)
|
|
{
|
|
this.services.getTaggingService().addTags(this.nodeRef, Arrays.asList(tags));
|
|
updateTagProperty();
|
|
}
|
|
|
|
/**
|
|
* Removes a tag from the node
|
|
*
|
|
* @param tag tag name
|
|
*/
|
|
public void removeTag(String tag)
|
|
{
|
|
this.services.getTaggingService().removeTag(this.nodeRef, tag);
|
|
updateTagProperty();
|
|
}
|
|
|
|
/**
|
|
* Removes all the tags from the node
|
|
*
|
|
* @param tags array of tag names
|
|
*/
|
|
public void removeTags(String[] tags)
|
|
{
|
|
this.services.getTaggingService().removeTags(this.nodeRef, Arrays.asList(tags));
|
|
updateTagProperty();
|
|
}
|
|
|
|
/**
|
|
* Get all the tags applied to this node
|
|
*
|
|
* @return String[] array containing all the tag applied to this node
|
|
*/
|
|
public String[] getTags()
|
|
{
|
|
String[] result = null;
|
|
List<String> tags = this.services.getTaggingService().getTags(this.nodeRef);
|
|
if (tags.isEmpty() == true)
|
|
{
|
|
result = new String[0];
|
|
}
|
|
else
|
|
{
|
|
result = (String[])tags.toArray(new String[tags.size()]);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Set the tags applied to this node. This overwirtes the list of tags currently applied to the
|
|
* node.
|
|
*
|
|
* @param tags array of tags
|
|
*/
|
|
public void setTags(String[] tags)
|
|
{
|
|
this.services.getTaggingService().setTags(this.nodeRef, Arrays.asList(tags));
|
|
updateTagProperty();
|
|
}
|
|
|
|
private void updateTagProperty()
|
|
{
|
|
Serializable tags = this.services.getNodeService().getProperty(this.nodeRef, ContentModel.PROP_TAGS);
|
|
if (this.properties != null)
|
|
{
|
|
this.properties.put(ContentModel.PROP_TAGS.toString(), getValueConverter().convertValueForScript(ContentModel.PROP_TAGS, tags));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets whether this node is a tag scope or not
|
|
*
|
|
* @param value true if this node is a tag scope, false otherwise
|
|
*/
|
|
public void setIsTagScope(boolean value)
|
|
{
|
|
boolean currentValue = this.services.getTaggingService().isTagScope(this.nodeRef);
|
|
if (currentValue != value)
|
|
{
|
|
if (value == true)
|
|
{
|
|
// Add the tag scope
|
|
this.services.getTaggingService().addTagScope(this.nodeRef);
|
|
}
|
|
else
|
|
{
|
|
// Remove the tag scope
|
|
this.services.getTaggingService().removeTagScope(this.nodeRef);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Gets whether this node is a tag scope
|
|
*
|
|
* @return boolean true if this node is a tag scope, false otherwise
|
|
*/
|
|
public boolean getIsTagScope()
|
|
{
|
|
return this.services.getTaggingService().isTagScope(this.nodeRef);
|
|
}
|
|
|
|
/**
|
|
* Gets the 'nearest' tag scope to this node by travesing up the parent hierarchy untill one is found.
|
|
* <p>
|
|
* If none is found, null is returned.
|
|
*
|
|
* @return TagScope the 'nearest' tag scope
|
|
*/
|
|
public TagScope getTagScope()
|
|
{
|
|
TagScope tagScope = null;
|
|
org.alfresco.service.cmr.tagging.TagScope tagScopeImpl = this.services.getTaggingService().findTagScope(this.nodeRef);
|
|
if (tagScopeImpl != null)
|
|
{
|
|
tagScope = new TagScope(this.services.getTaggingService(), tagScopeImpl);
|
|
}
|
|
return tagScope;
|
|
}
|
|
|
|
/**
|
|
* Gets all (deep) children of this node that have the tag specified.
|
|
*
|
|
* @param tag tag name
|
|
* @return ScriptNode[] nodes that are deep children of the node with the tag
|
|
*/
|
|
public ScriptNode[] childrenByTags(String tag)
|
|
{
|
|
List<NodeRef> nodeRefs = this.services.getTaggingService().findTaggedNodes(this.nodeRef.getStoreRef(), tag, this.nodeRef);
|
|
ScriptNode[] nodes = new ScriptNode[nodeRefs.size()];
|
|
int index = 0;
|
|
for (NodeRef node : nodeRefs)
|
|
{
|
|
nodes[index] = new ScriptNode(node, this.services, this.scope);
|
|
index ++;
|
|
}
|
|
return nodes;
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Workflow methods
|
|
|
|
/**
|
|
* Get active workflow instances this node belongs to
|
|
*
|
|
* @return the active workflow instances this node belongs to
|
|
*/
|
|
public Scriptable getActiveWorkflows()
|
|
{
|
|
if (this.activeWorkflows == null)
|
|
{
|
|
WorkflowService workflowService = this.services.getWorkflowService();
|
|
|
|
List<WorkflowInstance> workflowInstances = workflowService.getWorkflowsForContent(this.nodeRef, true);
|
|
Object[] jsWorkflowInstances = new Object[workflowInstances.size()];
|
|
int index = 0;
|
|
for (WorkflowInstance workflowInstance : workflowInstances)
|
|
{
|
|
jsWorkflowInstances[index++] = new JscriptWorkflowInstance(workflowInstance, this.services, this.scope);
|
|
}
|
|
this.activeWorkflows = Context.getCurrentContext().newArray(this.scope, jsWorkflowInstances);
|
|
}
|
|
|
|
return this.activeWorkflows;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Site methods
|
|
|
|
/**
|
|
* Returns the short name of the site this node is located within. If the
|
|
* node is not located within a site null is returned.
|
|
*
|
|
* @return The short name of the site this node is located within, null
|
|
* if the node is not located within a site.
|
|
*/
|
|
public String getSiteShortName()
|
|
{
|
|
if (!this.siteNameResolved)
|
|
{
|
|
this.siteNameResolved = true;
|
|
|
|
Path path = this.services.getNodeService().getPath(getNodeRef());
|
|
|
|
if (logger.isDebugEnabled())
|
|
logger.debug("Determing if node is within a site using path: " + path);
|
|
|
|
for (int i = 0; i < path.size(); i++)
|
|
{
|
|
if ("st:sites".equals(path.get(i).getPrefixedString(this.services.getNamespaceService())))
|
|
{
|
|
// we now know the node is in a site, find the next element in the array (if there is one)
|
|
if ((i+1) < path.size())
|
|
{
|
|
// get the site name
|
|
Path.Element siteName = path.get(i+1);
|
|
|
|
// remove the "cm:" prefix and add to result object
|
|
this.siteName = ISO9075.decode(siteName.getPrefixedString(
|
|
this.services.getNamespaceService()).substring(3));
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (logger.isDebugEnabled())
|
|
{
|
|
logger.debug(this.siteName != null ?
|
|
"Node is in the site named \"" + this.siteName + "\"" : "Node is not in a site");
|
|
}
|
|
|
|
return this.siteName;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Helper methods
|
|
|
|
/**
|
|
* Override Object.toString() to provide useful debug output
|
|
*/
|
|
public String toString()
|
|
{
|
|
if (this.nodeService.exists(nodeRef))
|
|
{
|
|
if (this.services.getPermissionService().hasPermission(nodeRef, PermissionService.READ_PROPERTIES) == AccessStatus.ALLOWED)
|
|
{
|
|
// TODO: DC: Allow debug output of property values - for now it's disabled as this could potentially
|
|
// follow a large network of nodes. Unfortunately, JBPM issues unprotected debug statements
|
|
// where node.toString is used - will request this is fixed in next release of JBPM.
|
|
return "Node Type: " + getType() + ", Node Aspects: " + getAspectsSet().toString();
|
|
}
|
|
else
|
|
{
|
|
return "Access denied to node " + nodeRef;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
return "Node no longer exists: " + nodeRef;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the JSON representation of this node.
|
|
*
|
|
* @param useShortQNames if true short-form qnames will be returned, else long-form.
|
|
* @return The JSON representation of this node
|
|
*/
|
|
public String toJSON(boolean useShortQNames)
|
|
{
|
|
// This method is used by the /api/metadata web script
|
|
String jsonStr = "{}";
|
|
|
|
if (this.nodeService.exists(nodeRef))
|
|
{
|
|
if(this.services.getPublicServiceAccessService().hasAccess(ServiceRegistry.NODE_SERVICE.getLocalName(), "getProperties", this.nodeRef) == AccessStatus.ALLOWED)
|
|
{
|
|
JSONObject json = new JSONObject();
|
|
|
|
try
|
|
{
|
|
// add type info
|
|
json.put("nodeRef", this.getNodeRef().toString());
|
|
|
|
String typeString = useShortQNames ? getShortQName(this.getQNameType())
|
|
: this.getType();
|
|
json.put("type", typeString);
|
|
json.put("mimetype", this.getMimetype());
|
|
|
|
// Fetch all properties
|
|
Map<QName, Serializable> nodeProperties = this.nodeService.getProperties(this.nodeRef);
|
|
|
|
// Do any special conversion steps that are needed
|
|
for (QName longQName : nodeProperties.keySet())
|
|
{
|
|
Serializable value = nodeProperties.get(longQName);
|
|
|
|
if (value instanceof Date)
|
|
{
|
|
value = ISO8601DateFormat.format((Date)value);
|
|
nodeProperties.put(longQName, value);
|
|
}
|
|
if (value instanceof NodeRef)
|
|
{
|
|
value = ((NodeRef)value).toString();
|
|
nodeProperties.put(longQName, value);
|
|
}
|
|
}
|
|
|
|
if (useShortQNames)
|
|
{
|
|
Map<String, Serializable> nodePropertiesShortQNames
|
|
= new LinkedHashMap<String, Serializable>(nodeProperties.size());
|
|
for (QName nextLongQName : nodeProperties.keySet())
|
|
{
|
|
try
|
|
{
|
|
String nextShortQName = getShortQName(nextLongQName);
|
|
nodePropertiesShortQNames.put(nextShortQName, nodeProperties.get(nextLongQName));
|
|
}
|
|
catch (NamespaceException ne)
|
|
{
|
|
// ignore properties that do not have a registered namespace
|
|
|
|
if (logger.isDebugEnabled())
|
|
logger.debug("Ignoring property '" + nextLongQName + "' as it's namespace is not registered");
|
|
}
|
|
}
|
|
json.put("properties", nodePropertiesShortQNames);
|
|
}
|
|
else
|
|
{
|
|
json.put("properties", nodeProperties);
|
|
}
|
|
|
|
// add aspects as an array
|
|
Set<QName> nodeAspects = this.nodeService.getAspects(this.nodeRef);
|
|
if (useShortQNames)
|
|
{
|
|
Set<String> nodeAspectsShortQNames
|
|
= new LinkedHashSet<String>(nodeAspects.size());
|
|
for (QName nextLongQName : nodeAspects)
|
|
{
|
|
String nextShortQName = getShortQName(nextLongQName);
|
|
nodeAspectsShortQNames.add(nextShortQName);
|
|
}
|
|
json.put("aspects", nodeAspectsShortQNames);
|
|
}
|
|
else
|
|
{
|
|
json.put("aspects", nodeAspects);
|
|
}
|
|
}
|
|
catch (JSONException error)
|
|
{
|
|
error.printStackTrace();
|
|
}
|
|
|
|
jsonStr = json.toString();
|
|
}
|
|
}
|
|
|
|
return jsonStr;
|
|
}
|
|
|
|
/**
|
|
* Returns the JSON representation of this node. Long-form QNames are used in the
|
|
* result.
|
|
*
|
|
* @return The JSON representation of this node
|
|
*/
|
|
public String toJSON()
|
|
{
|
|
return this.toJSON(false);
|
|
}
|
|
|
|
/**
|
|
* Given a long-form QName, this method uses the namespace service to create a
|
|
* short-form QName string.
|
|
*
|
|
* @param longQName
|
|
* @return the short form of the QName string, e.g. "cm:content"
|
|
*/
|
|
protected String getShortQName(QName longQName)
|
|
{
|
|
return longQName.toPrefixString(services.getNamespaceService());
|
|
}
|
|
|
|
/**
|
|
* Helper to create a QName from either a fully qualified or short-name QName string
|
|
*
|
|
* @param s Fully qualified or short-name QName string
|
|
*
|
|
* @return QName
|
|
*/
|
|
protected QName createQName(String s)
|
|
{
|
|
QName qname;
|
|
if (s.indexOf(NAMESPACE_BEGIN) != -1)
|
|
{
|
|
qname = QName.createQName(s);
|
|
}
|
|
else
|
|
{
|
|
qname = QName.createQName(s, this.services.getNamespaceService());
|
|
}
|
|
return qname;
|
|
}
|
|
|
|
/**
|
|
* Reset the Node cached state
|
|
*/
|
|
public void reset()
|
|
{
|
|
this.name = null;
|
|
this.type = null;
|
|
this.properties = null;
|
|
this.aspects = null;
|
|
this.targetAssocs = null;
|
|
this.sourceAssocs = null;
|
|
this.childAssocs = null;
|
|
this.children = null;
|
|
this.hasChildren = null;
|
|
this.parentAssocs = null;
|
|
this.displayPath = null;
|
|
this.isDocument = null;
|
|
this.isContainer = null;
|
|
this.parent = null;
|
|
this.primaryParentAssoc = null;
|
|
this.activeWorkflows = null;
|
|
this.siteName = null;
|
|
this.siteNameResolved = false;
|
|
}
|
|
|
|
/**
|
|
* @return helper object to perform cross repository copy of JavaScript Node objects
|
|
*/
|
|
protected CrossRepositoryCopy getCrossRepositoryCopyHelper()
|
|
{
|
|
return (CrossRepositoryCopy)this.services.getService(
|
|
QName.createQName("", CrossRepositoryCopy.BEAN_NAME));
|
|
}
|
|
|
|
/**
|
|
* Return a list or a single Node from executing an xpath against the parent Node.
|
|
*
|
|
* @param xpath XPath to execute
|
|
* @param firstOnly True to return the first result only
|
|
*
|
|
* @return Object[] can be empty but never null
|
|
*/
|
|
private Object[] getChildrenByXPath(String xpath, QueryParameterDefinition[] params, boolean firstOnly)
|
|
{
|
|
Object[] result = null;
|
|
|
|
if (xpath.length() != 0)
|
|
{
|
|
if (logger.isDebugEnabled())
|
|
{
|
|
logger.debug("Executing xpath: " + xpath);
|
|
if (params != null)
|
|
{
|
|
for (int i=0; i<params.length; i++)
|
|
{
|
|
logger.debug(" [" + params[i].getQName() + "]=" + params[i].getDefault());
|
|
}
|
|
}
|
|
}
|
|
|
|
List<NodeRef> nodes = this.services.getSearchService().selectNodes(this.nodeRef, xpath, params,
|
|
this.services.getNamespaceService(), false);
|
|
|
|
// see if we only want the first result
|
|
if (firstOnly == true)
|
|
{
|
|
if (nodes.size() != 0)
|
|
{
|
|
result = new Object[1];
|
|
result[0] = newInstance(nodes.get(0), this.services, this.scope);
|
|
}
|
|
}
|
|
// or all the results
|
|
else
|
|
{
|
|
result = new Object[nodes.size()];
|
|
for (int i = 0; i < nodes.size(); i++)
|
|
{
|
|
NodeRef ref = nodes.get(i);
|
|
result[i] = newInstance(ref, this.services, this.scope);
|
|
}
|
|
}
|
|
}
|
|
|
|
return result != null ? result : new Object[0];
|
|
}
|
|
|
|
/**
|
|
* Helper to return true if the supplied property value is a ScriptContentData object
|
|
*
|
|
* @param o Object to test
|
|
*
|
|
* @return true if instanceof ScriptContentData, false otherwise
|
|
*/
|
|
public boolean isScriptContent(Object o)
|
|
{
|
|
return (o instanceof ScriptContentData);
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Value Conversion
|
|
|
|
/**
|
|
* Gets the node value converter
|
|
*
|
|
* @return the node value converter
|
|
*/
|
|
protected NodeValueConverter getValueConverter()
|
|
{
|
|
if (converter == null)
|
|
{
|
|
converter = createValueConverter();
|
|
}
|
|
return converter;
|
|
}
|
|
|
|
/**
|
|
* Constructs the node value converter
|
|
*
|
|
* @return the node value converter
|
|
*/
|
|
protected NodeValueConverter createValueConverter()
|
|
{
|
|
return new NodeValueConverter();
|
|
}
|
|
|
|
// Set support
|
|
|
|
/**
|
|
* Value converter with knowledge of Node specific value types
|
|
*/
|
|
public class NodeValueConverter extends ValueConverter
|
|
{
|
|
/**
|
|
* Convert an object from any repository serialized value to a valid script object. This includes converting
|
|
* Collection multi-value properties into JavaScript Array objects.
|
|
*
|
|
* @param qname QName of the property value for conversion
|
|
* @param value Property value
|
|
*
|
|
* @return Value safe for scripting usage
|
|
*/
|
|
public Serializable convertValueForScript(QName qname, Serializable value)
|
|
{
|
|
return convertValueForScript(services, scope, qname, value);
|
|
}
|
|
|
|
/*
|
|
* (non-Javadoc)
|
|
*
|
|
* @see org.alfresco.repo.jscript.ValueConverter#convertValueForScript(org.alfresco.service.ServiceRegistry,
|
|
* org.mozilla.javascript.Scriptable, org.alfresco.service.namespace.QName, java.io.Serializable)
|
|
*/
|
|
@Override
|
|
public Serializable convertValueForScript(ServiceRegistry services, Scriptable scope, QName qname,
|
|
Serializable value)
|
|
{
|
|
if (value instanceof ContentData)
|
|
{
|
|
// ContentData object properties are converted to ScriptContentData objects
|
|
// so the content and other properties of those objects can be accessed
|
|
value = new ScriptContentData((ContentData) value, qname);
|
|
}
|
|
else
|
|
{
|
|
value = super.convertValueForScript(services, scope, qname, value);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
/*
|
|
* (non-Javadoc)
|
|
*
|
|
* @see org.alfresco.repo.jscript.ValueConverter#convertValueForRepo(java.io.Serializable)
|
|
*/
|
|
@Override
|
|
public Serializable convertValueForRepo(Serializable value)
|
|
{
|
|
if (value instanceof ScriptContentData)
|
|
{
|
|
// convert back to ContentData
|
|
value = ((ScriptContentData) value).contentData;
|
|
}
|
|
else
|
|
{
|
|
value = super.convertValueForRepo(value);
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
|
|
|
|
// ------------------------------------------------------------------------------
|
|
// Inner Classes
|
|
|
|
|
|
/**
|
|
* Inner class wrapping and providing access to a ContentData property
|
|
*/
|
|
public class ScriptContentData implements Content, Serializable
|
|
{
|
|
private static final long serialVersionUID = -7819328543933312278L;
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param contentData The ContentData object this object wraps
|
|
* @param property The property the ContentData is attached too
|
|
*/
|
|
public ScriptContentData(ContentData contentData, QName property)
|
|
{
|
|
this.contentData = contentData;
|
|
this.property = property;
|
|
}
|
|
|
|
/* (non-Javadoc)
|
|
* @see org.alfresco.repo.jscript.ScriptNode.ScriptContent#getContent()
|
|
*/
|
|
public String getContent()
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentReader reader = contentService.getReader(nodeRef, property);
|
|
return (reader != null && reader.exists()) ? reader.getContentString() : "";
|
|
}
|
|
|
|
/*
|
|
* (non-Javadoc)
|
|
* @see org.springframework.extensions.surf.util.Content#getInputStream()
|
|
*/
|
|
public InputStream getInputStream()
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentReader reader = contentService.getReader(nodeRef, property);
|
|
return (reader != null && reader.exists()) ? reader.getContentInputStream() : null;
|
|
}
|
|
|
|
/*
|
|
* (non-Javadoc)
|
|
* @see org.springframework.extensions.surf.util.Content#getReader()
|
|
*/
|
|
public Reader getReader()
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentReader reader = contentService.getReader(nodeRef, property);
|
|
|
|
if (reader != null && reader.exists())
|
|
{
|
|
try
|
|
{
|
|
return (contentData.getEncoding() == null) ? new InputStreamReader(reader.getContentInputStream()) : new InputStreamReader(reader.getContentInputStream(), contentData.getEncoding());
|
|
}
|
|
catch (IOException e)
|
|
{
|
|
// NOTE: fall-through
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Set the content stream
|
|
*
|
|
* @param content Content string to set
|
|
*/
|
|
public void setContent(String content)
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentWriter writer = contentService.getWriter(nodeRef, this.property, true);
|
|
writer.setMimetype(getMimetype()); // use existing mimetype value
|
|
writer.putContent(content);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* Set the content stream from another content object.
|
|
*
|
|
* @param content ScriptContent to set
|
|
*/
|
|
public void write(Content content)
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentWriter writer = contentService.getWriter(nodeRef, this.property, true);
|
|
writer.setMimetype(content.getMimetype());
|
|
writer.setEncoding(content.getEncoding());
|
|
writer.putContent(content.getInputStream());
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* Set the content stream from another content object.
|
|
*
|
|
* @param content ScriptContent to set
|
|
* @param applyMimetype If true, apply the mimetype from the Content object, else leave the original mimetype
|
|
* @param guessEncoding If true, guess the encoding from the underlying input stream, else use encoding set in
|
|
* the Content object as supplied.
|
|
*/
|
|
public void write(Content content, boolean applyMimetype, boolean guessEncoding)
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentWriter writer = contentService.getWriter(nodeRef, this.property, true);
|
|
InputStream is = null;
|
|
if (applyMimetype)
|
|
{
|
|
writer.setMimetype(content.getMimetype());
|
|
}
|
|
if (guessEncoding)
|
|
{
|
|
is = new BufferedInputStream(content.getInputStream());
|
|
is.mark(1024);
|
|
writer.setEncoding(guessEncoding(is, false));
|
|
try
|
|
{
|
|
is.reset();
|
|
}
|
|
catch (IOException e)
|
|
{
|
|
}
|
|
}
|
|
else
|
|
{
|
|
writer.setEncoding(content.getEncoding());
|
|
is = content.getInputStream();
|
|
}
|
|
writer.putContent(is);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* Set the content stream from another input stream.
|
|
*
|
|
* @param inputStream
|
|
*/
|
|
public void write(InputStream inputStream)
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentWriter writer = contentService.getWriter(nodeRef, this.property, true);
|
|
writer.putContent(inputStream);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* Delete the content stream
|
|
*/
|
|
public void delete()
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentWriter writer = contentService.getWriter(nodeRef, this.property, true);
|
|
OutputStream output = writer.getContentOutputStream();
|
|
try
|
|
{
|
|
output.close();
|
|
}
|
|
catch (IOException e)
|
|
{
|
|
// NOTE: fall-through
|
|
}
|
|
writer.setMimetype(null);
|
|
writer.setEncoding(null);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* @return download URL to the content
|
|
*/
|
|
public String getUrl()
|
|
{
|
|
return MessageFormat.format(CONTENT_PROP_URL, new Object[] { nodeRef.getStoreRef().getProtocol(),
|
|
nodeRef.getStoreRef().getIdentifier(), nodeRef.getId(),
|
|
URLEncoder.encode(getName()),
|
|
URLEncoder.encode(property.toString()) });
|
|
}
|
|
|
|
/**
|
|
* @return download URL to the content for a document item only
|
|
*/
|
|
public String getDownloadUrl()
|
|
{
|
|
if (getIsDocument() == true)
|
|
{
|
|
return MessageFormat.format(CONTENT_DOWNLOAD_PROP_URL, new Object[] {
|
|
nodeRef.getStoreRef().getProtocol(),
|
|
nodeRef.getStoreRef().getIdentifier(),
|
|
nodeRef.getId(),
|
|
URLEncoder.encode(getName()),
|
|
URLEncoder.encode(property.toString()) });
|
|
}
|
|
else
|
|
{
|
|
return "";
|
|
}
|
|
}
|
|
|
|
public long getSize()
|
|
{
|
|
return contentData.getSize();
|
|
}
|
|
|
|
public String getMimetype()
|
|
{
|
|
return contentData.getMimetype();
|
|
}
|
|
|
|
public String getEncoding()
|
|
{
|
|
return contentData.getEncoding();
|
|
}
|
|
|
|
public void setEncoding(String encoding)
|
|
{
|
|
this.contentData = ContentData.setEncoding(this.contentData, encoding);
|
|
services.getNodeService().setProperty(nodeRef, this.property, this.contentData);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
public void setMimetype(String mimetype)
|
|
{
|
|
this.contentData = ContentData.setMimetype(this.contentData, mimetype);
|
|
services.getNodeService().setProperty(nodeRef, this.property, this.contentData);
|
|
|
|
// update cached variables after putContent()
|
|
this.contentData = (ContentData) services.getNodeService().getProperty(nodeRef, this.property);
|
|
}
|
|
|
|
/**
|
|
* Guess the mimetype for the given filename - uses the extension to match on system mimetype map
|
|
*/
|
|
public void guessMimetype(String filename)
|
|
{
|
|
ContentService contentService = services.getContentService();
|
|
ContentReader reader = contentService.getReader(nodeRef, property);
|
|
setMimetype(services.getMimetypeService().guessMimetype(filename, reader));
|
|
}
|
|
|
|
/**
|
|
* Guess the character encoding of a file. For non-text files UTF-8 default is applied, otherwise
|
|
* the appropriate encoding (such as UTF-16 or similar) will be appiled if detected.
|
|
*/
|
|
public void guessEncoding()
|
|
{
|
|
setEncoding(guessEncoding(getInputStream(), true));
|
|
}
|
|
|
|
private String guessEncoding(InputStream in, boolean close)
|
|
{
|
|
String encoding = "UTF-8";
|
|
try
|
|
{
|
|
if (in != null)
|
|
{
|
|
Charset charset = services.getMimetypeService().getContentCharsetFinder().getCharset(in, getMimetype());
|
|
encoding = charset.name();
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
try
|
|
{
|
|
if (close && in != null)
|
|
{
|
|
in.close();
|
|
}
|
|
}
|
|
catch (IOException ioErr)
|
|
{
|
|
}
|
|
}
|
|
return encoding;
|
|
}
|
|
|
|
private ContentData contentData;
|
|
private QName property;
|
|
}
|
|
|
|
/**
|
|
* Interface contract for simple anonymous classes that implement document transformations
|
|
*/
|
|
private interface Transformer
|
|
{
|
|
/**
|
|
* Transform the reader to the specified writer
|
|
*
|
|
* @param contentService ContentService
|
|
* @param noderef NodeRef of the destination for the transform
|
|
* @param reader Source reader
|
|
* @param writer Destination writer
|
|
*
|
|
* @return Node representing the transformed entity
|
|
*/
|
|
ScriptNode transform(ContentService contentService, NodeRef noderef, ContentReader reader, ContentWriter writer);
|
|
}
|
|
|
|
|
|
/**
|
|
* NamespacePrefixResolverProvider getter implementation
|
|
*/
|
|
public NamespacePrefixResolver getNamespacePrefixResolver()
|
|
{
|
|
return this.services.getNamespaceService();
|
|
}
|
|
} |