mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-22 15:12:38 +00:00 
			
		
		
		
	48730: Record only merge from 4-1.BUG-FIX (4.1.5) to HEAD-BUG-FIX (4.2)
      48447: Performance problem with Eclipse can be fixed by excluding the web-client build directory from the searchable resources.
         Done for 58 other projects (generally include in the 4.1 code line)
      48726: Reverse Merge (did not work after an Eclipse Clean) - already done on HEAD
         48447: Performance problem with Eclipse can be fixed by excluding the web-client build directory from the searchable resources.
         Done for 58 other projects (generally include in the 4.1 code line)
   48929: Synced up with HEAD r48903
   48957: Merged PATCHES/V4.1.4 to HEAD-BUG-FIX
      47847: Merged V4.1-BUG-FIX to PATCHES/V4.1.4
         47625: Fixed  ALF-18063
         - AlfrescoUtil's getPages method now parses json using jsonUtils.toObject to force strict json
         - CSRFFilter got new throwError action meaning the filter now can stop certain resources form being reached directly from the browser
         - /remoteadm/* webscripts can no longer be accessed directly from the browser
         - Modified regexps to be easier to read
         - Added config samples
      47879: Merged V4.1-BUG-FIX to PATCHES/V4.1.4
         47878: Final part of fix for ALF-14086 - Sort order of folders including hyphens ( - ) are different in folder-tree and view on folders (in Share)
          - People Finder
      47912: Merged BRANCHES/DEV/V4.1-BUG-FIX to PATCHES/V4.1.4:
         47909: ALF-18257: partial fix - adds back in missing TinyMCE files.
      47928: Merged V4.1-BUG-FIX to PATCHES/V4.1.4
         47913: Final part of fix for ALF-18257 - AWE panel is not displayed
      47932: ALF-14587: Created article or publication cant be viewed on WQS site
      - Fix by Dmitry Vaserin
      - Same fix was done to style1.get.html.ftl in ALF-17117
      47956: ALF-10243: workflow shows wrong date
      - Missing WQS date field configuration
      47983: ALF-15803: CH, NL, RU: There are no Chinese, Russian, Dutch templates for IMAP messages and Chinese, Russian templates for Email messages
      - Email templates from Gloria
      48026: ALF-15803: CH, NL, RU: There are no Chinese, Russian, Dutch templates for IMAP messages and Chinese, Russian templates for Email messages
      - Email templates from Gloria
      48027: Merged V4.1-BUG-FIX to PATCHES/V4.1.4
         48020: Fix for ALF-18220 - Cancel checkout in 'View Original Document' view fails and causes document stuck in checkedout state.
      48048: Merged BRANCHES/DEV/DWEBSTER/HEAD_AWE to PATCHES/V4.1.4:
         48047: Fixes: ALF-14758. Adds z-index as a work around to an overlapping menu bug that occurs if they are created in a particular order and viewed on a large enough resolution monitor.
      48086: Merged BRANCHES/DEV/V4.1-BUG-FIX to PATCHES/V4.1.4:
         48083: Fixes ALF-14931
      48121: Merged BRANCHES/DEV/V4.1-BUG-FIX to PATCHES/V4.1.4:
         48116: Fixes build failure caused by 'fix' for ALF-14931
      48153: Merged V4.1-BUG-FIX to PATCHES/V4.1.4
         48102: Fix for ALF-18354 - SPP-Meeting workspace event can be edited and deleted on Agenda tab
         48124: Fixes ALF-18349: Slideshare and YouTube channels can now be created.
         48151: Fix for ALF-15475 - Liferay Portal: IE9: Layout of portlets is broken when document details page is opened
      48156: Improved fix (removed useless media queries) for ALF-18196 - Document Preview - Flash Movie Sized to a Thin Horizontal Bar at 1024x768 Resolution
      48185: Another go at a solution for ALF-16413 - Share asks for Basic-Auth while not needed trying to access RSS feeds (thus breaking SSO).
      48189: ALF-17818 - Strikethrough formatting is not preserved in Wiki
      48208: ALF-18385: Alfresco Explorer: Localization is not applied after login.
      - Fixes regression caused by ALF-17189
      48261: Fix for ALF-18352 - Manager can not update permissions more than once for the folder/content created by other user
      48264: Merged DEV to PATCHES/V4.1.4
         48258: ALF-18325 : Can't delete wcmqs if it was published to live
         We should also check for site is not being deleted before creating site container.
      48283: Correction to fix for ALF-18196
      48284: Fix for ALF-18328 - Share dependencies containing dependencies breaks Hazelcast clustering
      48352: Fix for MNT-6390 - Incorrect type for Group and Review workflow after task approved by all users
      48354: Fix for MNT-3196 - No information is displayed in My Activities and Site Activities dashlets for content creation
      48390: MNT-6292: Created article or publication cant be viewed on WQS site
      - Fix researched by Dmity Vaserin
      - More handling of empty lists
      - It seems that WCMQS requires renditions to be indexed (in order for them to be returned by CMIS queries) so we must remove the ASPECT_INDEX_CONTROL aspect from them in RenditionHelper
      48424: Fixed MNT-7521 "Document Library" instead "File Plan" displayed in the site navigation bar
      48430: MNT-7522: Users with "read only" or "read and file" permissions don't have access to RM folders
      - In ScriptNode.childByNamePath() check for PermissionService.READ permission rather than PermissionService.READ_PROPERTIES permission, which apparently isn't compatible with RM!
      48434: MNT-7522: Users with "read only" or "read and file" permissions don't have access to RM folders
      - Corrected fix with help from Andy Hind
      48500: Fixed MNT-6311 "Activities with Google Docs are not displayed in My Site Activities and Site Activities dashlets"
      48548: MNT-7528 My Tasks to Do and My Pooled Tasks dashlets do not show the correct description of a workflow 
      48635: MNT-8482 Enterprise 4.1.4: Backwards compatibility breaking change in public Java API class org.alfresco.service.cmr.dictionary.ClassDefinition
         - Added back in missing methods and using StaticMessageLookup so will still have the issue identified by MNT-413.
           The new methods still exist and don't have this issue.
      48645: MNT-8484: Oracle: An unexpected difference is present after comparison of database schema with reference schema (post-upgrade) 
      48729: MNT-8498 Could not transform .psd file 
      48745: MNT-8527: Activity Feed Cleaner broken on Oracle
      - Due to ibatis weirdnesses, the select_activity_user_feeds_greater_than_max query has to exclude results with null user_ids (non user feeds) as this maps to a null row on Oracle and results in an NPE!
      - Also fixed all comparisons with feed_user_id to be NULL safe
      48751: Fixed MNT-8504 "Script error occurs when clicking Edit Tags action from document/folder details page"
      - Removed js_string encoding of nodeRef and made sure nodeRef is taken from the repo instead of the page url
      48752: MNT-8467: Cannot authorize to Share Kerberos automatically
      - Reversed r44754 performance fix causing this (Sorry Kev)
      48755: MNT-8514: Incorrect workflow status is displayed on My Tasks dashlet
      - Fix by Viachaslau reviewed by Frederik
      48756: MNT-8531: Merged PATCHES/V4.1.3 to PATCHES/V4.1.4
         48683: MNT-8494: Lucene index recovery fails to reindex an orphaned node whose parent is deleted
         - Added unit test for this and fixed regression over MNT-6545 fix
         48717: MNT-8494: Lucene index recovery fails to reindex an orphaned node whose parent is deleted
         - Strengthened unit test to check for deleted nodes with parents and strengthened handling of these as a result
         - Hopefully fixes unit test failures
      48798: Fixed MNT-7646 "CLONE - [Pentest] Internet Explorer <= v8 XSS"
      - Share now inspects text/xml content from the webscript content, if it finds a <!DOCTYPE svg> it will change the mimetype to text/plain.
      48810: Upgrade Surf version in POM files to 1.2.0-M4 (r1217)
      48819: Fixed MNT-8504 "Script error occurs when clicking Edit Tags action from document/folder details page" part 2
      - Fix on folders page
      48820: Make SchemaReferenceFileTest actually output its reason for failing - I can't track down the temp files!
      48822: MNT-8508: Fixing description-property in LazyActivitiWorkflowTask which is not the same as WorklfowTask.description
      48823: BDE-130: fail the build when yuicompressor minimization fails
      48890: Fixed MNT-8554 "FF: Download button on Document Details page is not working"
      - Note also fixes the "Comment" widget in the header not working (in any browser)
      48933: MNT-8560: It's impossible to find the user by any property except the username
      Modify People.getSortedPeopleObjects() method to create ArrayList based on provided List<NodeRef> and use it for sort, because provided List<NodeRef> can be unmodifiable list.
   48964: MNT-6494: Fixed merge issue (already fixed in HEAD-BUG-FIX)
   49033: Merged PATCHES/V4.1.4 to HEAD-BUG-FIX-QA
      48996: MNT-8564 : CLONE - Incorrect saving process when editing event
      Fix event.put.json.ftl to produce valid json
      49006: MNT-8555: Reverse Merge MNT-6419 (ALF-17089)
         45245: ALF-17089 (Displaying Url Name instead of site Name in Select form)
         47089: ALF-17089 (Displaying Url Name instead of site Name in Select form)
   49039: Merged HEAD to HEAD-BUG-FIX-QA
      48931: Fixed CLOUD-1511 "My Documents dashlet is crashing when inspecting the syncmode value"
      48955: Fix name of wdr-deployment jar in mvninstall task
      48979: ALF-18573: fix sharepoint module context name.
      This is a change that was appropriate for cloud but had crept in to HEAD during the megamerge.
      48988: Fix integration with CLOUD2 + compile and package remote-api test classes separately
      48995: CLOUD-1460 "CloudConv: Re-implement header customizations"
      - #1. Re-add link so network admin can access admin console 
         Added in widget ids that could be referenced to add and remove widgets in the user & app items menu
      49008: Fix public api tests
      49014: Fix public api tests
   49136: Fix public api tests
   49141: Merged HEAD-BUG-FIX to HEAD-QA (thus completing 4.1.4 -> HEAD-QA merge)
      49139: Merged PATCHES/V4.1.4 to BRANCHES/DEV/HEAD-BUG-FIX (mergeinfo committed earlier)
         48735: Fixed MNT-8506 & MNT-8507
            * MNT-8506 "A server error has occurred when press Follow/Unfollow button from User Profile"
            * MNT-8507 "A server error occurs when set "Private" checkbox on My Profile"
            * Instead of using Alfresco.util.Ajax.request and creating XHR requests forms were being submitted 
              meaning the Alfresco-CSRFToken never got added.
            * Now refactored to use Alfresco.util.Ajax.request and post directly to the proxy, 
              meaning the Share "services" now can be removed.
         48812: MNT-8506 / MNT-8507: Fix up by Vasily to avoid YUI compression error
         48813: Fixed MNT-8507 "A server error occurs when set "Private" checkbox on My Profile" part 2
            - variable used keyword "private", now refactored to "isPrivate"
   49154: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      48470: Merged BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAPI3 to BRANCHES/DEV/CLOUD1:
           48041: Merged BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAP2 to BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAPI3:
                47129: Merged BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAPI1 to BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAP2:
                     45439: PUBLICAPI-39: "Favourites API" wip
                     45681: PUBLICAPI-39: "Favourites API" wip
                     45897: Test a fix for issue found during bm testing - CMIS: null entries in property list values
                     45965: Public api performance investigation: test node bulk loading change
                     45978: PUBLICAPI-39/40: wip, changes from review.
                     46019: PUBLICAPI-39: "Favourites API" wip
                     46056: Re-instate opencmis config that was mistakenly changed
                     46156: Public api performance investigation: test node bulk loading change
                     46193: PUBLICAPI-39: "Favourites API" wip
                     46200: PUBLICAPI-39: "Favourites API" wip
                     46247: Upgrade to OpenCMIS 0.8
                     46248: PUBLICAPI-72: "Add users role to Site entity"
                     46249: PUBLICAPI-72: "Add users role to Site entity"
                     46250: PUBLICAPI-40: "Site membership API" - modifiedAt timestamp + some tidy up
                     46251: PUBLICAPI-40: "Site membership API" - bug fixes
                     46253: public api tests refactoring (split tests into separate classes by functional area), site membership api tests
                     46273: public api tests refactoring (split tests into separate classes by functional area), site membership api tests
                     46298: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46299: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46300: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46311: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46337: PUBLICAPI-79: "GET a Favorite"
                     46338: PUBLICAPI-79: "GET a Favorite"
                     46339: public api tests: some refactoring
                     46340: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46345: Added name parameter to the @uniqueId annotation. e.g. @UniqueId(name="targetGuid")
                     46346: PUBLICAPI-39: "Favourites API" - tests
                     46359: PUBLICAPI-40: "Site membership API" - PUT requests and bug fixes
                     46380: PUBLICAPI-40: "Site membership API" - PUT requests and bug fixes
                     46381: PUBLICAPI-40: "Site membership API" - PUT requests and bug fixes
                     46397: PUBLICAPI-40: "Site membership API" - add missing file
                     46411: public api tests: re-instate tests (accidentally commented out)
                     46427: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46442: Cloud OpenCMIS browser binding support through the OpenCMIS Alfresco webscript (currently disabled)
                     46467: PUBLICAPI-80: "GET on a deleted Favorite answers a 500"
                     46476: Upgrade to OpenCMIS 0.8
                     46514: Initial implementation of WHERE clause functionality. (Likely to change a lot).
                     46518: PUBLICAPI-75: "Use a 'where' parameter to specify criteria for favorite result set content "
   49157: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      48552: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/CLOUD1
            48551: Fix for     DEVOPS-2976   P1 - Solr nodes crash with tracking enabled 
                   + configuration fixes
      48796: Merged BRANCHES/DEV/FEATURES/ACCOUNT_TYPES to BRANCHES/DEV/CLOUD1:
           48092: Resolve CLOUD-1303: Add 2 new Account Types to Cloud
           48325: Translations from Gloria
           48445: Japanese: translation update from Gloria
           48698: JA: Translation update from Gloria.
      49130: Merged BRANCHES/DEV/CLOUD1-HOTFIX to BRANCHES/DEV/CLOUD1:
           49120: Merged BRANCHES/DEV/FEATURES/CLOUD1-PUBLICAPI3 to BRANCHES/DEV/CLOUD1-HOTFIX:
                49093: CLOUD-1518: "Favourite sites are not displayed correctly"
                49110: CLOUD-1518: "Favourite sites are not displayed correctly": fix up PreferencesService to return correct site preference keys
                49116: CLOUD-1518: "Favourite sites are not displayed correctly": favourite sites uses the favourites service rather thanpreferences service directly, making it and favourites consistent
   49159: fixup pesky solrcore.properties
   49197: Fix for Cloud AccountUsageQuotaTest - make action-executor node service mt aware (fallout from previous commit that now does a node existence check in action executor that needs to be mt aware)
   49199: Fix tests
   49241: CLOUD-1527 "Logo in Share footer is broken"
   - Added /res to make logo getting displayed when running in multi tenancy mode
   49250: CLOUD-1527 "Logo in Share footer is broken"
   - Removed image dimensions to make image replaceable
   49267: Fix tests: remove incorrect OpenCMIS jars
   49272: Use newer patched version of chemistry-opencmis-server-bindings
   49273: Removed extraneous chemistry library dependency
   49293: CLOUD-1512 "SAML integration needs to override "CSRFPolicy" config"
   - Adding new (overridable) getPath() method in CSRFFilter
   49315: Merged PATCHES/V4.1.4 to HEAD-QA
      49236: Fixed MNT-8615 "Browser page is not reloaded after deleting folder/file or after creation content in Repository"
      49301: Fixed MNT-8615 "Browser page is not reloaded after deleting folder/file or after creation content in Repository" part 2
   49316: Merged PATCHES/V4.1.4 to HEAD-QA (RECORD ONLY)
      48735: Fixed MNT-8506 & MNT-8507
         * MNT-8506 "A server error has occurred when press Follow/Unfollow button from User Profile"
         * MNT-8507 "A server error occurs when set "Private" checkbox on My Profile"
         * Instead of using Alfresco.util.Ajax.request and creating XHR requests forms were being submitted meaning the Alfresco-CSRFToken never got added.
         * Now refactored to use Alfresco.util.Ajax.request and post directly to the proxy, meaning the Share "services" now can be removed.
      48812: MNT-8506 / MNT-8507: Fix up by Vasily to avoid YUI compression error
      48813: Fixed MNT-8507 "A server error occurs when set "Private" checkbox on My Profile" part 2
      - variable used keyword "private", now refactored to "isPrivate"
   49328: Fix up People script class so that it deals with access denied exceptions (this will hopefully fix failing PeopleRestApiTest in Cloud2)
   49346: Fixed CLOUD-1532 "Test email signup page is brokebn due to new header"
   49364: Fixed CLOUD-1524 "Unable to retrieve Google Docs subsystem status from Alfresco (re: googleEditable not defined ?)"
   49373: CONV: Fix CLOUD-1524 / CLOUD-446 - disable GoogleDocsV1 subsystem and prevent deferred start (and associated error) when Share later calls explicit "status" webscript (to see that it is disabled)
   - revert 49364 and replace with alternative fix 
   - see also similar IMAP status issue (CLOUD-1507)
   49375: CONV: Fix CLOUD-1507 -fix IMAP subsystem to prevent deferred start (and associated error) when Share later calls explicit "status" webscript (to see that it is disabled)
   - see also similar GDocsV1 status issue (CLOUD-1524 / CLOUD-446)
   49416: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      49303: CLOUD-1518: "Favourite sites are not displayed correctly": fix fallout
   49418: CONV: Fix ALF-18698 - Core MT: "create tenant" ignores the tenant-specific root contentstore dir path (if specified)
   49433: Fix for failing Cloud2 test https://bamboo.alfresco.com/bamboo/browse/THOR-CLOUD2HEADQA-JOB1-38/test/case/12496544
   49491: CLOUD-1552 "The invite link icon in the new header's site title "area" points to the "on premise" invite page (which has been admin secured) instead of the invite dialog"
   - Improving extensibility by adding further widgets ids for header menu
   49590: CLOUD-1556 "CloudConv: CLOUD2 does not start on DP: "Possible CSRF attack"" part 1
   - Not a bug in the CSRFFIlter but an improvement that introduces a <properties> elements which may hold variables used by the CSRFPolicy config 
      to avoid repetition and the possibility to only override the <properties> section in different environemnts, i.e. set different referer proxies in test & prod.
   49758: Fix for CLOUD-1542, Google Docs action was missing from the action list in the document library.
   An action is still listed in the document-browse action group that has no definition, this was causing the web tier webscript to abort processing of any actions defined after it i.e. any custom actions.
   49789: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      49788: L10N update from Gloria, based on EN r48822
   49801: Move CLOUD2 to HEAD-QA, as part of Alfresco One project
   49806: Bring POM file back from CLOUD2
   49807: Fix order of from/set/to lines in rules, to match schema validation.
   49829: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      49808: CLOUD-1615: CloudConv - Regression: CLONE: the info panel isn't wide enough to display the share link without a line break
           - Increased panel and image widths
   49830: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      49823: CLOUD-1609: CloudConv - Regression: DAM: Item header was moved (also incorrect color is using) for folders and files.
           - Added overrides and changes to gallery view to accomodate changed CSS in detail and simple view
   49831: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      49827: CLOUD-1610: CloudConv - Regression: DAM: More menu does not disappear after first opening more link
           - Fixed el passed in for onEventUnhighlightRow in fnHideDetailPanel
           - Also added hiding of more actions on mouseleave of more actions
   49834: SVN ignore patterns - "test" folder.
   49835: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA
      49800: Fix to build.properties that got munged in rev 49209. Fix to share-header that was horribly merged to pull in code with dependencies that aren't present yet. Fix to login dialog width that hasn't made it over to this branch for some reason.
   49872: Added name to logging of transformer registration
   49885: CLOUD-1626: CloudConv - Regression: Networks are not displayed via WebDav.
   Reinstated code that was originally part of THOR1_SPRINTS (r34168) but was lost as part of convergence work.
   49889: CLOUD-1669: upgrade jibx from 1.2.3 to 1.2.5 to allow full JDK7 support
   49906: Merged BRANCHES/DEV/BELARUS/HEAD-QA-2013_04_22 to BRANCHES/DEV/HEAD-QA:
      49794: ALF-18797 : SPP: Check out action doesn't work for Document Workspace
   Merged but with minor bug fix.
   49908: CLOUD-1649: Added the missing network menu for the public users.
   49916: Fix to User Factory bean setter to avoid:
   WARN  [springframework.beans.GenericTypeAwarePropertyDescriptor] [main] Invalid JavaBean property 'userFactory' being accessed! Ambiguous write methods found next to actually used
   49917: Merged SHOULDERS to HEAD-QA
      48984: Minor UI fixes:
                   - fix height of user status text-area to be consistent across browsers
                   - fix hover style and font size of user status text-area
                   - fix login dialog width since Cloud merge
   49918: Fix for CLOUD-1571 - CloudConv - Regression: My Sites dashet isn't loaded
   49919: Fix for CLOUD-1648 - CloudConv - regression: cloud2headqa is not localized in any language
    - code for the login language drop-down refactored to fit within login component rather than global function javascript
    - fixed to work since code convergence also
   49920: Fix for CLOUD-1584 - CloudConv - Regression: Some dashlets should be absent on User/Site Dashboard
    - Disabled new dashlets (Site Search, Saved Search, My Discussions) because they are too exciting for the Cloud.
   49921: CLOUD-1669 Switch -source and -target compilation parameters to 1.7
   49923: Fix to 404 error on login page (visible in firebug etc.)
   Correct list of image dependencies.
   49924: Fix for CLOUD-1629 - CloudConv - Regression: It's impossible to remove the Welcome Widget on Site Dashboard
    - encoding of preference values must handle "." dot character or it ends up as nested preference!
   49927: Fix for CLOUD-1577 - CloudConv - regression:Incorrect display of status message
   49928: Fix for CLOUD-1586 - CloudConv - regression:The user is redirected to login page after clicking 'Invite' button from Site members page
    - Button should be disabled on Cloud (redirect is because on-premise Invite page is admin only!) - but pattern to do so has changed since merge with 4.2.
   49931: WebDAV: exclusive lock could be obtained when shared lock already in place.
   In addition, the invalid lock state that was produced then led to a stack overflow due to infinite loop.
   49933: Added svn:ignore entires for build/pom.xml and root/privatemodules/thor/config/log4j.properties
   49936: CLOUD-1600: CloudConv - regression: Cannot add a tag through inline edit option
      - Removed mandatory validator
   49938: Merge V4.1-BUG-FIX to HEAD-QA
      48904: BDE-109: generate coverage statistics from JaCoCo when running tests
      48925: 3rd party classpath refers to available mysql-connector-java-5.1.13-bin.jar
      49065: Limit coverage to org.alfresco classes, otherwise overhead is too big and tests too slow 
      49474: MNT-8668 Remove Maven declared dependency on addressing-1.0.mar
      49655: BDE-109 new Ant task to generate JaCoCo report for test coverage
   49941: CLOUD-1668: CloudConv - Regression: Incorrect displaying of default avatar on task section on document's details page
      - Fixed avatar link
   49943: CLOUD-1661: CloudConv - Regression: progress bar is absent in Storage section on Account Summary page
      - Brought theme-bg-3, theme-bg-4, and theme-border-5 CSS over from greyTheme to lightTheme
   49958: SpringSurf libs upgrade to r1257
   Updated Cloud share config overrides to remove values that are now the default in Surf (e.g. reconnect-timeout etc.)
   49965: Fix for CLOUD-1588 - CloudConv - Regression: "Download as Zip" option should be absent
    - remove the lovely new Download as Zip feature from Cloud as it is Officially Too Exciting.
   49975: Fix for CLOUD-1663 - CloudConv - Regression: Search query containing wildcards is partially cut off on the results page (encoding is not being applied)
   49977: CLOUD-1640: Fixed external user site member visibility issue.
   50007: Fix to disallow change of own user role or removal in Site Member for the current user.
   50011: Activiti schema reference files ignore table column ordering (testing CLOUD-1675)
   50012: Reorganize repository database bootstrap beans to register scripts and patches with SchemaBootstrap (CLOUD-1675)
    - New Spring config file: alfresco/dbscripts/db-schema-context.xml
    - Split Repo, JPBM, AVM and Activiti scripts
   50013: Update module properties to reflect latest Alfresco stack requirement and set version to 2.0
   50014: Fixed CLOUD-1675: Upgrade failed from CLOUD1-BUG-FIX3 to HEAD-QA
    - Use module-friendly registration of create scripts
    - Note: Schema validation will report errors due to 2 Cloud-specific tables in the 'alf' schema namespace
   50018: CLOUD-1568: CloudConv - regression:created/modified time is not displayed in Properties section
      - Added unit test for combining default controls where a new data type is added, which was failing
      - Fixed DefaultControlsConfigElement.combine method
   50023: Fix for CLOUD-1578 - CloudConv - regression: Unable to join a site
   Fix for CLOUD-1579 - CloudConv - Regression: It's impossible to perform "Leave Site" action by Site Member from Site Dashboard
   50027: BDE-150 Create RPMs for Cloud deployment
   50028: ALF-18908: WebDAV litmus tests failing on HEAD-QA/CLOUD2
   50032: CLOUD-1624: CloudConv - regression: Login button is missing on Quick Share page
      - Moved CSS loading to quickshare/header.get.html.ftl
   50039: CLOUD-1626: networks not displayed via WebDAV (fix for PROPFIND)
   50040: CLOUD-1637 / MNT-8462 "Post" button works incorrectly
      - Added check for response type of string to userStatusUpdateSuccess and userStatusUpdateFailure since it's already coming back as a JSON object
   50047: Add missing dependency on java-property-utils-1.6
   50048: BDE-150: fix RPMs
   50051: Fix for CLOUD-1664 - CloudConv - Regression: New logo is not applied
   Fix for CLOUD-1580 - CloudConv - Regression: It's impossible to change the application theme
    - Fix to correctly handle "network admin" concept - manually allow the network admin to perform admin actions where appropriate - AFAIK this can never have worked correctly... remove nonsense runas=admin webscripts overrides that don't work anyway.
    - Clean up of related scripts.
   50055: CLOUD-1607: filter sent-invites-min.js coming from Enterprise build, so that it gets overridden by the non-minimised cloud one
   50070: Temporarily adding exploded Ant build for cloud2 environment. 
   This now means the same steps for building a cloud1 environment can be used to build a cloud2 environment.
   NOTE: There is one outstanding licence issue to resolve that will be fixed in a subsequent commit.
   50088: Remove unused or duplicate MyBatis config (CLOUD-1672)
   50090: CLOUD-1672: CloudConv: MyBatis error when downloadCleanerJob is run
    - Cloud was duplicating the MyBatis configuration
    - Created Cloud-specific SQL templates
    - Tested using sign ups and general use but full account and tenant usage needs to be tested
   50092: CLOUD-1591: CloudConv - Regression: Incorrect layout of "Link to Rule Set" page
      - Made the more specific rules-picker treeview width override lightTheme's overriding treeview width
   50094: Fixes for exploded Ant build.
   50097: Added ignore of projects/web-client/source/web/images/logo/alfresco_enterprise.gif which is apparently moved from elsewhere during some build step
   50099: CLOUD-1614: CloudConv - Regression: Incorrect label "Start workflow" displays instead of "Create Task"
      - Added page.startWorkflow.title to cloud localization files
   50101: BDE-154: upgrade urlrewritefilter to 4.0.4, avoiding compilation warning
   50102: CLOUD-1602: CloudConv - Regression - Chrome specific: Close (X) button doesn't work on "Alfresco Cloud information" window
      - Removed committed minified about-share-min.js and ant touch command which kept it in place
   50107: Merged HEAD-BUG-FIX to HEAD-QA (4.2) for CLOUD-1466: stop relying on a patched version of wss4j
      49973: Merged DEV to HEAD-BUG-FIX (4.2)
         49648: MNT-8652 Treat PasswordText as default for CMIS Web Service authentication (using patched version of wss4j)  
               Execute PasswordTypeInterceptor before WSS4JInInterceptor to add PasswordText type as default if Password element misses it. Add system test for case. 
      50076: Follow-up on MNT-8652: rename CXFAuthenticationTest into CXFAuthenticationSystemTest, so that it is not executed in web-service-client tests, but only in system-build-test
   50109: Fix to issue where application logo was resized and broke aspect ratio of uploaded image. Also changed size to valid for new header.
   50113: Fix for CLOUD-1576 - CloudConv - Regression: Incorrect notification is displayed when Site Admin performs "Leave Site" action
   50114: Fix for CLOUD-1662 - CloudConv - Regression: Incorrect behavior when searching with empty field
    - trim searchbox text before event publication
   50116: Cleanup of copy/pasted files...
   50117: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/HEAD-QA
      47811: ALF-18245 - BM-0013: Soak: Run 06: Search population of ScriptNode is expensive:
             Refactor Share search UI and REST API to provide server-side paging.
   50122: MNT-8480 - cloud sync and revert
   50126: BDE-150 Add build number as RPM release property
   50150: Merged V4.1-BUG-FIX (4.1.5) to HEAD-QA (4.2)
      << MNT-8721 Security fix from 4.1.4.2 >>
      << Merge direct from V4.1-BUG-FIX rather than HEAD-BUG-FIX as r49633 was merged with a block of other changes >>
      49633: Merged DEV to V4.1-BUG-FIX (4.1.5)
         49574: MNT-8473: Share's workflow-details page exposes all workflow instances to everyone
         Introduce WorkflowPermissionInterceptor that check if current user allowed to perform operation.
      49752: MNT-8473: Share's workflow-details page exposes all workflow instances to everyone 
         Fix errors that were found during unit testing.
   50153: ALF-18708: Merged V4.1-BUG-FIX to HEAD-QA
      << Security fix from 4.1.4.4 >>
         50151: MNT-8866: Merge V4.1.4 (4.1.4.4) to V4.1-BUG-FIX (4.1.5)
            49802: MNT-8725: CLONE - Security context for Alfresco Explorer is not being cleaned up after processing a request
               JSF phase listener, which cleans up security context for Alfresco Explorer requests. It is necessary for avoiding sharing of security context between Alfresco Explorer and CMIS services, based on OpenCMIS libraries
            49803: MNT-8725: CLONE - Security context for Alfresco Explorer is not being cleaned up after processing a request
               Configuration part of the fix has been reversed
            49804: MNT-8725: CLONE - Security context for Alfresco Explorer is not being cleaned up after processing a request
               JSF phase listener, which cleans up security context for Alfresco Explorer requests. It is necessary for avoiding sharing of security context between Alfresco Explorer and CMIS services, based on OpenCMIS libraries
   50164: Merged V4.1-BUG-FIX to HEAD-QA
      << Priority for transformer.remote.alfresco. Includes JMX controls for the Transformers subsystem >>
      49038: MNT-7158 Merge DEV to HEAD-BUF-FIX (4.2)
         - Addition of TransformerConfigMBeam (JMX bean) to control transformers dynamically
         - Transformer level config limits are now a default for mimetype specific values. Possible now that we don't have these in the Spring config.
           previously had been 'combined' which was not too clear. Also added system wide mimetype defaults.
         - All transformers are now 'known' to the content transformer registry - not just the ones that may be selected. i.e it knows about the component transformers.
         - TransformerSelector now round robins all equal priority transformers until they reach their thresholds, rather than doing so for each one in turn.
      49135: MNT-7158 Investigating adding priorities to transformers
         - Changes to make it possible to check debugTransformer.txt output in 4.1.5 against 4.2
      49190: MNT-7158 fix build failure testGetMaxSourceSizeKBytesPageSupports
      49207: MNT-7158 Investigating adding priorities to transformers
         - Modified priorities so that it is now deterministic
           Try as sourceMimetype of 00 in JMX op getTransformationsByExtension to double check
         - Changed default priorities 5 and 10 to 50 and 100 for EXPLICIT and DEFAULT to give more room in the future
         - Combined transformer.complex.AdobeIllustrator.Image with transformer.complex.PDF.Image as they were the same
           except for the transformer settings
         - TransformerDebug now uses [---] as the priority for compound transformers
         - TransformerDebug now includes Component in the type of compound transformers e.g. fred<<ComplexCompound>>
      49947: MNT-7158 Investigating adding priorities to transformers
         - Added limits based on use. So a limit may be set for "index", "doclib", "webpreview", "syncRule", "asyncRule" ...
      49985: MNT-7158 Investigating adding priorities to transformers
         - Allow <tomcat>/shared/classes/alfresco-global.properties to be used in transformers.
      50002: MNT-7158 Investigating adding priorities to transformers
         - Missing file
      50061: MNT-7158 Investigating adding priorities to transformers
         - Set up remote transformations 
            BinaryPassThrough.priority=20
            remote.alfresco.priority=30
            remoteServer.priority=40
      50143: MNT-7158 Investigating adding priorities to transformers
         - Show default values in getProperties, comment out values set to defaults and add under line transformer name
   50166: Merged V4.1-BUG-FIX to HEAD-QA
      << Two 4.1.4.4 hot fixes >>
      50165: Merged V4.1-BUG-FIX (4.1.5) to HEAD-BUG-FIX (4.2)
         50110: MNT-8844 - CSRFFilter disables https login
         50158: MNT-8870: Merge V4.1.4 (4.1.4.4) to V4.1-BUG-FIX (4.1.5)
            50120: MNT-8858: AJAX Servlet does not clean up security context after processing a request
               Modification for cleaning up security context after processing an AJAX request to 'org.alfresco.web.app.servlet.ajax.AjaxServlet'
   50169: CLOUD-1641: Made people-finder to use CannedQuery rather than Solr.
   50173: Fix for CLOUD-1587 - CloudConv - Regression: Advanced Search link should be unavailable.
   50174: Fix for CLOUD-1608 - CloudConv - Regression: Changing roles of site members is displayed only after page refresh
   Fix for ALF-18534 - Changing roles of site members is displayed only after page refresh
   50176: Fix for CLOUD-1592 - CloudConv - Regression: Content I'm Editing dashlet contains redundant info (about blogs, wiki, forum)
   50177: CLOUD-1689: CloudConv - Regression:Networks are not displayed via SPP
   50182: ALF-18702 : Tenant isn't able to log in Share once another tenant has logged in before.
   Fix for clearing tenant user domain and authentication context after the Repository webscript is executed.
   Copied the TenantLeakLogger from CLOUD to HEAD into org.alfresco.web.app.servlet package.
   50183: CLOUD-1606, CLOUD-1459: Fixed the permissions issue for the get preferences in the MailActionExecuter.
   50184: Changed default Cloud Share Hazelcast config to empty IP list rather than promiscuous multicast - which was much too saucy!
   50185: XSS attack hardening - incorrect use of ?js_string for safe HTML element IDs - should be ?html
   50197: Code tidy
   50200: ALF-18957: SchemaReferenceFileTest failing on Oracle
   Updated ACT schema-reference file for Oracle.
   50203: Fix for CLOUD-1613 - CloudConv - Regression: CLONE: Under selected Item menu the manage permission should be hidden
   50208: Fix CLOUD-1660 - CloudConv - Regression: Incorrect layout of tag icon on Search result page
   50211: Fix to (unreported?) issue where Colleagues dashlet would not show the user status relative time - showed empty value in brackets ()
   50215: DEVOPS-3172, CLOUD-1701: cloud2/HEAD-QA unable to startup in DP environment.
   50227: CLOUD-1612: CloudConv - Regression: Incorrect location of help icon for "Message" field on "Create Task" page
      - Moved rendering of help element
      - Added CSS for desired positioning
   50245: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      50240: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/HEAD-BUG-FIX:
           50221: Merged DEV to V4.1-BUG-FIX (4.1.5)
                50030: Attempt a workaround for MNT-8704: WebDAV:Content does not disappear after being deleted
                 - Deprecate various methods related to the existing WebDAV use of sys:hidden
                 - Put in a timer task on use of the sys:hidden aspect in WebDAV DeleteMethod
                50093: MNT-8704: WebDAV:Content does not disappear after being deleted
                - "if (!fileFolderService.isHidden(nodeRef))" block was moved to deleteHackRunAs.doWork().
                50187: MNT-8704: added node existence check; unhiding node prior to deletion.
                As per Derek's comment:
                https://issues.alfresco.com/jira/browse/MNT-8704?focusedCommentId=227689&page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel#comment-227689
   50251: CLOUD-1702: Fixed NPE in WorkflowPermissionInterceptor, also fixed the issue that when the invitee tries to click on the link in the invitation email, 'org.alfresco.service.cmr.security.NoSuchPersonException' is thrown.
   50298: Fixed the failing unit tests as a result of the newly added WorkflowPermissionInterceptor.
   50306: Fix to merge causing ALF-18979 - Activiti servlet references on by default in web.xml
   Removed duplicated class now that the LoggerFilter is part of code webclient code
   50311: MNT-8890: Commented the newly added method in TenantRepositoryContainer in order to fix the failing unit tests.
   50318: Generate a jar of tests for alfresco-cloud, for consumption by QA tests later
   50322: Fix for CLOUD-1657 - CloudConv - Regression: Admin section in the Header should be located in the user's menu having name "Account settings"
   50335: ALF-18702 : Tenant isn't able to log in Share once another tenant has logged in before.
   Fixed failing JUnit tests.
   Reverted 50311 revision.
   50337: CLOUD-1699: Cloud example content thumbnails - some have the wrong width and look rubbish in doclib/search results
      - Added new versions of doclib for Data Sheets to sampleContent.acp
      - Added doclib and imgpreview for all content in sampleContent.acp except 'Alfresco Cloud Intro.mp4'
   50345: ALF-18702 : Tenant isn't able to log in Share once another tenant has logged in before.
   Fixed failing JUnit tests.
   50352: Fix for CLOUD-1691 - Incorrect notification when leaving name field empty through inline edit
    - fixes the issues where; length validator not fired, order of validators wrong for cm:name inline editor, fixes issue where previously validation would still be shown on reedit of field after Cancel << hat tip to Mr Draper for validation solution
   50353: Fix for CLOUD-1604 - CloudConv - regression: Links on "Oops page" should be highlited.
    - also removed duplicate messages and cleaned up message text from the late 90's.
   50356: Fixed CLOUD-1697: URL rewrite filter upgrade causes additional logging
   50357: Fixed the failing unit tests which were failing as a result of rev: 50335 commit.
   50374: Revert r50182 and the subsequent changes made to fix the fallout: r50311, r50335, r50345, and r50357
   50379: CLOUD-1669: update Eclipse JibX launchers with the upgraded JibX 1.2.5 jars
   50390: Revert the part of r50306 that updated enterprise-web-scripts-application-context.xml and deleted TenantLeakLogger.java.
   This was necessary following the reverts that took place at r50374.
   50396: Corrected name of content transform test class file.
   50397: Added new "transform" folder (which should have been there before)
   50398: Move content transform test class file into the new "transform" folder.
   50401: - Added two Eclipse projects: one for the thor private module ("Alfresco Cloud") and one for its dependencies ("Cloud Dependencies").
   - Turned the JiBX builder on by default in the data-model project.
   50410: MNT-8890: split classpath when executing JUnit tests in Ant, so that repository tests do not get classes of upstream projects
   50411: MNT-8890 follow-up
    - make enterprise repository tests run with a reduced classpath too, otherwise EntRepoCacheStopTest fails
    - add wdrdeployment classes to reduced classpath, otherwise enterprise tests cannot find alfresco/bootstrap/deployedSpace.xml
   50416: Removed unused component (no longer required in 4.2 so the override isn't required)
   50417: Dialog styling tweaks
   50422: Merged BRANCHES/DEV/FEATURES/CLOUD1-SECURITY to BRANCHES/DEV/HEAD-QA
      50409: Fix for CLOUD-1721
   50473: CLOUD-1726 CLOUD-1732: move tests CMISChangeLogServiceTest and SubsystemsTest from repository to remote-api, because they depend on resources in remote-api (follow-up of MNT-8890)
   50474: CLOUD-1736 Shouldn't have tweaked the classpath of system-build-test, since it already has its own (follow-up of MNT-8890)
   50479: ALF-18927: SPP: Incorrect information is displayed on Event Information window for yearly recurrect event
   BYMONTHDAY shouldn't be deleted for YEARLY recurrence.
   Also setting interval was not accurate.
   50480: ALF-18927: SPP: Incorrect information is displayed on Event Information window for yearly recurrect event
   Additional check in.
   50485: Fixes ALF-18702: Corrected the logic for how tenant switching is achieved so that both Enterprise and Cloud scenarios are catered for.
   50495: Fix for ALF-18981 - Non-ASCII characters are not displayed in Share interface correctly
    - Fix to deal with issue where a valid JSON result string is not always valid JavaScript text e.g. some unicode characters etc.
   50498: svn ignore patterns
   50499: files should not be in svn
   50500: remove eclipse folder settings from svn
   50505: Fix to (unreported?) search results paging issue - merge error meant that only the first page of results would ever be returned no matter what page was selected, but who goes past the first page anyway eh?
   50516: CLOUD-1718: "CloudConv - Regression:"Imap Attachments" and "IMAP Home" folders are displayed via CMIS Workbench client"
   50517: CLOUD-1727: CloudConv - Regression - WebDav: The mapped network drive could not be created on Windows
   This was broken by MNT-2823.
   50524: CLOUD-1665: New accounts are not available for CLOUD2
   Manually merged revision 48796 as it was mistakenly marked as merged.
   50525: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      50033: (RECORD ONLY) Merge HEAD-QA to CLOUD1 (for BDE-109: generate coverage on unit tests)
           Merge V4.1-BUG-FIX to HEAD-QA
              48904: BDE-109: generate coverage statistics from JaCoCo when running tests
              48925: 3rd party classpath refers to available mysql-connector-java-5.1.13-bin.jar
              49065: Limit coverage to org.alfresco classes, otherwise overhead is too big and tests too slow 
              49474: MNT-8668 Remove Maven declared dependency on addressing-1.0.mar
              49655: BDE-109 new Ant task to generate JaCoCo report for test coverage
   50582: Removed and ignored .classpath and .project
   50614: CLOUD-1713 append the Cloud build number after the Enterprise build number, as in: r50345-b65-c129
   Upgrade a few Maven plugins
   50649: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      50083: Merged BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3 to BRANCHES/DEV/CLOUD1:
           49357: New Cloud Bug Fix Branch
           49363: Merged BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX2 to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3:
                46487: Fixes CLOUD-1293: "Other" included in total activities count. Refactor activity groupings.
                46488: Fixes: CLOUD-1310: Adds "no-wrap" to last element to that the second link all appears on the same line.
                46493: Fixes: CLOUD-1229: Removes max-width CSS constraint so that table cells can expand to find room for long transltions.
                46535: PUBLICAPI-63: "Unable to list comments associated to a document : Malformated comment object ?": fix + tests
                46555: PUBLICAPI-90: "You can not remove yourself from a site using the -me- identifier"
                46826: Fix for CLOUD-1368 - Tomcat thread blocking during concurrent user testing causing service to become unresponsive.
                47249: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX2
                     47243: Fix for ALF-18194 - Bad performance with multiple contentstores browsing files on the slower stores
                47681: Merged HEAD to CLOUD1-BUG-FIX2
                     47448,47484,47504,47518,47552
                47865: GERMAN: Latest translation updates, based on EN r46485 & including fixes from check report.
                47866: FRENCH: Latest translation updates, based on EN r46485 & including fixes from check report.
                47867: SPANISH: Latest translation updates, based on EN r46485 & including fixes from check report.
                47868: ITALIAN: Latest translation updates, based on EN r46485 & including fixes from check report.
                47869: JAPANESE: Latest translation updates, based on EN r46485 & including fixes from check report.
                47885: ALL LANG: Minor update to translations to bring them in line with EN r47485
                48355: CLOUD-1458: Added MailActionExecuterTest to ActionTestSuite.
                48402: CLOUD-1458: Remove inadvertently added test class
                48561: CLOUD-1458, CLOUD-1459, CLOUD-1479, CLOUD-1485 - Fix MailActionExecuter to correctly resolve primary domain for sender and recipients.
                47249: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX2
                     47243: Fix for ALF-18194 - Bad performance with multiple contentstores browsing files on the slower stores
                47681: Merged HEAD to CLOUD1-BUG-FIX2
                     47448,47484,47504,47518,47552
                47865: GERMAN: Latest translation updates, based on EN r46485 & including fixes from check report.
                47866: FRENCH: Latest translation updates, based on EN r46485 & including fixes from check report.
                47867: SPANISH: Latest translation updates, based on EN r46485 & including fixes from check report.
                47868: ITALIAN: Latest translation updates, based on EN r46485 & including fixes from check report.
                47869: JAPANESE: Latest translation updates, based on EN r46485 & including fixes from check report.
                47885: ALL LANG: Minor update to translations to bring them in line with EN r47485
                48355: CLOUD-1458: Added MailActionExecuterTest to ActionTestSuite.
                48402: CLOUD-1458: Remove inadvertently added test class
                48561: CLOUD-1458, CLOUD-1459, CLOUD-1479, CLOUD-1485 - Fix MailActionExecuter to correctly resolve primary domain for sender and recipients.
           49371: PUBLICAPI-166: "Unable to upload large file on Cloud  (> 5MB)"
           49389: Merged BRANCHES/DEV/FEATURES/CLOUD1_SAML1 to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3:
                49295: CLOUD-1529: Made SessionIndex optional in the SSO response. Also commented out 'autoProvisionEnabled' and 'alfrescoLoginCredentialEnabled' from the SAMLConfigSettings as they are not implemented yet.
           49406: CLOUD-1488: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3 (Also added the missing elements from the merge):
                ALF-16480: Merged PATCHES/V4.1.1 to V4.1-BUG-FIX
                 43252: MNT-166: Document lock not removed after the lock expiry date has been reached
                     -  Changed evaluator "evaluator.doclib.metadata.isLocked" to use "evaluator.doclib.indicator.nodeLocked" and "evaluator.doclib.indicator.lockOwner" evaluator.
                 43253: MNT-165: "Cancel Editing" does not completely remove lock from document
                     - Added documentLibrary cancel editing action for locked documents.
                 43300: MNT-171: Merged V4.1-BUG-FIX to PATCHES/V4.1.1 (modified)
                 43096: Fix for ALF-16283 - When document is checked out, 'Edit Online' and 'Upload New Version' options should not be visible on the original document.
                 43311: MNT-165: "Cancel Editing" does not completely remove lock from document
                     - Fix for page refresh problem when cancel editing on details page
                 43421: MNT-186: 4.1.1.7 HF: Webscipt error on doclib page, containing locked by other users files
                     - Change evaluator.doclib.metadata.isLocked to break circular dependency
                 43755: MNT-202: Upload New Version not available for a document that has been edited offline
                     - Upload New Version is now available if editable by user (respecting locks, type of checkout, etc).
           49463: CLOUD-1510 - Cache remote transformation calls to isTransformable
               - Make TransformationOptions serializable
               - Expose ehcache statistics and configuration, via JMX
           49492: CLOUD-1510 - Fix broken unit test
           49631: CLOUD-1455: "Hide activity in a site from the activity feed"
           49662: CLOUD-1455: "Hide activity in a site from the activity feed": fix broken unit test
           49759: Merged BRANCHES/DEV/HEAD-QA to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3:
                49758: Fix for CLOUD-1542, Google Docs action was missing from the action list in the document library.
           49814: CLOUD-1455: "Hide activity in a site from the activity feed"
           49866: CLOUD-1597: Added the missing evaluators.
           49948: Merged BRANCHES/DEV/FEATURES/CLOUD1_GOOGLEDOCS to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3:
                49880: Update Google Docs integration from 2.0.2 to 2.0.3.
           50045: Merged BRANCHES/DEV/FEATURES/CLOUD1_GOOGLEDOCS to BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIX3:
                50043: Update 2.0.3 AMPs
   50686: GERMAN: Latest translations and updates based on EN rev49789. Jira fixes: MNT-7526, ALF-18650, ALF-18523, ALF-18521, ALF-18255, ALF-18069
   50689: SPANISH: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255
   50691: FRENCH: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255
   50693: ITALIAN: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255
   50694: JAPANESE: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255
   50695: DUTCH: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255
   50696: RUSSIAN: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255, ALF-18504, ALF-18503, ALF-18502, ALF-17793
   50699: CHINESE: Latest translations and updates based on EN rev49789. Jira fixes: ALF-18650, ALF-18523, ALF-18521, ALF-18255, MNT-8874, MNT-8874, MNT-8874
   50709: CLOUD-1699: Cloud example content thumbnails - some have the wrong width and look rubbish in doclib/search results
      - Fixed imgpreview for Alfresco Mobile.pdf
   50711: Merged DEV to HEAD-QA (4.2.0)
      50482: ALF-18996 : Upgrade from 3.4.12 to 4.2.0 fails: ERROR [domain.schema.SchemaBootstrap] [localhost-startStop-1] Statement execution failed: SQL: ALTER TABLE ACT_RU_TASK MODIFY ASSIGNEE_ VARCHAR(255) Error: Table 'alfupg.ACT_RU_TASK' doesn't exist
         - Correctly handle activiti tables creation/upgrade when performing upgrade from 3.4.x.
   50728: CLOUD-1754 do not fail the build as soon as tests fail: we are now quarantining tests in Bamboo, so installers need to be built regardless
   50755: Merged HEAD-BUG-FIX to HEAD-QA
       50754: Merged V4.1-BUG-FIX (4.1.6) to HEAD-BUG-FIX
          50753: Merged DEV to V4.1-BUG-FIX (4.1.6)
             50727: MNT-3368: BM-0013: Soak: Run 08: OnPremiseSyncPullJob runs without a transaction
              - Add transactions. 
              50750: MNT-3368: BM-0013: Soak: Run 08: OnPremiseSyncPullJob runs without a transaction
              - Correction for 'readOnly' parameter. 
   50759: Merged BRANCHES/DEV/JAMAL/Fast_Test into HEAD-QA (CLOUD-1680: Make thor tests run blazingly fast)
      50635: CLOUD-1680: Make thor tests run blazingly fast.
      50684: CLOUD-1680: Regouped a few of the public api tests.
      50688: CLOUD-1680: Regouped a few of the public api tests.
      50706: CLOUD-1680: Moved the excludedGroups from the parent pom to thor. Also grouped the webdav tests.
      50710: CLOUD-1680: Deleted the un-necessary junit category.
   50773: Fixing up the tests to separate out the cloud tests.  MailActionExecuter should now get the locale correctly.
   50779: Redundant file
   50780: Redundant File
   50782: Redundant File
   50783: Fixed missing double quote
   50784: Missed strings included
   50785: Missed key included
   50795: ALF-19035 - MT Cloud Policies run with inconsistent "short" base and "full" node refs.
   50823: ALF-17548: Disable Google Docs v1 integration
     * MBean controlling the subsystem has been removed (commented out) from bootstrap-context.xml
     * Admin console web script org/alfresco/components/console/repository/google-docs.get has been removed
     * Admin console form config has been deleted from share-form-config.xml
     * No changes have been made to the installer, as I could find no evidence that the configuration of the v1 subsystem was being changed by it
     * Note: The subsystem property 'googledocs.googleeditable.enabled' was already set to false.
   50825: ALF-17548: Fix status web script which assumed the presence of the Google Docs v1 MBean removed in r50823. Now the web script returns enabled: false if the bean is not present.
   50827: CLOUD-1600: CloudConv - regression: Cannot add a tag through inline edit option
      - Added ignoreEmpty argument to accommodate changes in r50352
   50831: CLOUD-1770: Redundant Perform Action ("Embed properties as metadata in content") is displayed for Content Rules
      - Moved embedder action to sample file since we don't supply any embedders out-of-the-box
   50833: CLOUD-1770: Redundant Perform Action ("Embed properties as metadata in content") is displayed for Content Rules
      - Programmatically created executer rather than loading from context
   50838: Cloud: Patch overrides
    - Moved live patches to 'module-schema-context.xml' ('patch.redeployProcessesWithReminders2' and 'patch.redeployProcessesWithReminders3')
    - No Sample Web Site in Cloud: (CLOUD-1480) Created no-op patch for 'patch.siteLoadPatch.swsdp'
    - IMAP is disabled in Cloud: (CLOUD-1743): Created no-op patch for 'patch.imapFolders'
   50839: ALF-18702: Tenant isn't able to log in Share once another tenant has logged in before
     Following review, added logic in AuthenticationServiceImpl.authenticate that is similar to AuthenticationServiceImpl.validate in order to handle case where the tenant is determined by the requested URI
   50851: First part of fix for ALF-19052.
   50853: ALF-18096: Integrate document security AMP files from the Alfresco Mobile iOS 1.5 project
     * New aspect dp:restrictable
     * Form configuration for properties provided by this aspect
     * Note: As detailed on the ticket the source files have been copied into the repository and slingshot project and mofified there.
   50864: Fix for CLOUD-1769 CloudConv - Admin section in header menu instead of Account Settings are available for free account user
   50879: ALF-18914 : SPP: It's possible to create meeting with empty subject field linking to an existing workspace
   Added a check for an empty subject when creating a meeting from iCal.
   50922: Revert r50853 mistakenly adding Mobile project dp:restricted aspect to HEAD-QA
   50925: CLOUD-1759: Fixed most of the Sonar ‘Critical’ violations.
   50937: CloudConv - Regression - CLONE - No notification is displayed when performing "Leave Site" action
   50938: Added Cloud override to remove MyFiles and Shared links from toolbar
   50941: Fix for     CLOUD-1768   CloudConv - Regression - CLONE: It's impossible to find the user by any property except the username
      and          CLOUD-1778   CloudConv:in people search first name and last name people search always return zero results
   50942: Fix for CLOUD-1741 - CloudConv: User should be redirected to Alfresco Cloud Documentation instead of Alfresco Community 4.2 Documentation. Comical spelling error in config.
   50945: GERMAN: Latest Cloud translations and updates based on EN rev50920. Jira fixes: CLOUD-1729
   50946: FRENCH: Latest Cloud translations and updates based on EN rev50920. Jira fixes: CLOUD-1729 and CLOUD-1763
   50947: SPANISH: Latest Cloud translations and updates based on EN rev50920. Jira fixes: CLOUD-1729 and CLOUD-1763
   50948: ITALIAN: Latest Cloud translations and updates based on EN rev50920. Jira fixes: CLOUD-1729 and CLOUD-1763
   50949: JAPANESE: Latest Cloud translations and updates based on EN rev50920. Jira fixes: CLOUD-1729 and CLOUD-1763
   50957: Fix for CLOUD-1623 Unfriendly warning appears when user try to create private site with the same URL
   50960: CLOUD-1772: Configuration files have been localized 
    - Removed all pointlessly translated configuration files
   50961: MNT-8480 : New web script for delete audit entry.
   50968: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      50624: (RECORD ONLY) Delete generation of installers in continuous, to allow 'amber' builds
   50973: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      50568: Merged BRANCHES/DEV/FEATURES/CLOUD1-SECURITY to BRANCHES/DEV/CLOUD1
            50423: Merged PATCHES/V4.1.1 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY
                      44129: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                      44130: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                      44129: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                   Merged PATCHES/V4.1.4 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY
                      45951: Fix for ALF-17687 BM-0013: Soak: Run 02: SolrJSONResultSet must preload nodes
      50622: Fix for CLOUD-1751   Users searching in their home tenant (or other person related operations) should not be subject to visibility checks
      50664: Fix for CLOUD-1751   Users searching in their home tenant (or other person related operations) should not be subject to visibility checks
   50976: CLOUD-1634: CloudConv - regression: Upload form is not closed automatically after upload limit error occurs
      - Added Javascript to hide upload status and show upload control again
   50992: Changing the modifiers for the "addAuthorityNameIfMatches" methods from "private" to "protected" to allow the RM module do RM specific changes without copying a lot of the existing code. 
   50993: Fix for ALF-18850 - Incorrect label "page.ruleEdit.title" on New Rule page from Repository Browser
   51031: CLOUD-1761: Activity is not generated when deleting document via SPP/WebDAV - fix part 1
   This is the first part of a fix: activities were not being raised due to MNT-181 fix. This is now fixed. However, the posts are not being processed properly, which also needs fixing.
   51083: Fixed the failing test (testHasSite)
   51085: First part of fix for CLOUD-1787
   51091: CLOUD-446: Remove 'overridden' googledocs subsystem bean from Cloud override context, which was causing the subsystem to start up still
   51092: Fix to URLRewrite config rule to correct escape UTF-8 characters either side of the detected '@' character when rewriting URLs where browsers have not correctly dealt with the @ character.
   51097: Updated BG for ludicrous screen resolution CLOUD-1795
   51098: Fix for CLOUD-1795 - Login dialog centre on window resize
   51103: Fix for CLOUD-1796 CloudConv:-Search pagination to other pages no results are shown
   51105: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
      51104: Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/HEAD-BUG-FIX:
           51101: MNT-8704: node was not being unhidden (thread local client marker needed setting)
   51107: Improvements to search result list handling - only populate node result structures for the items that are going to be returned.
   51140: Minor tweaks to search processing hand merged from 4.1.N
   51163: JAPANESE: Fix for CLOUD-1799
   51170: ALF-18074 add a new flag to enable clustering.
   51184: Make cloud tests depend on repository test resources, to be able to reuse them
   51234: CLOUD-1800: Made the Rules to be executed asynchronously in the cloud by overriding “rule-edit.js”.
   51245: Merged BRANCHES/DEV/HEAD42-MERGE to BRANCHES/DEV/HEAD-QA
      51230: Addition of a modified YUI Compressor ANT task to speed up the minimize-slingshot-deployed-javascript build step - from around 3 mins on most machines down to 3 secs. w00t
   AND THIS IS THE RESULT:
   ant clean incremental-webclient-tomcat-exploded incremental-slingshot-tomcat-exploded
   ...
   BUILD SUCCESSFUL
   Total time: 1 minute 18 seconds
   51248: Merged HEAD-BUG-FIX to HEAD-QA
      51247: Merged V4.1-BUG-FIX to HEAD-BUG-FIX (RECORD ONLY)
         51246 (RECORD ONLY): Merged PATCHES/V4.1.5 to V4.1-BUG-FIX
            51233: MNT-8129: NPE - Servlet.service() - getGlobalConfig for "webClientConfigService"
                Missing JGroups messages receiver has been added to handle cluster messages correctly.
                Port range default value has been increased to add some slack around required ports
         50999 (PARTIAL RECORD ONLY): Better logging for AsynchronouslyRefreshedCache (MNT-8129)
   51273: Partial fix for CLOUD-1806. This checkin fixes serialisation of TransformationSourceOptions objects.
   A subsequent check-in will fix deserialisation.
   51275: Fixed CLOUD-1804
   51279: Allow calls to NodeDAO.getParentAssocs to succeed for deleted nodes
    - The error message reported is incorrect.  Previously, no calls were made to the DAO to
      retrieve details for deleted nodes.  There was therefore an assumption that the node had been
      internally referenced and that this represented an error.  This is no longer the case (nor has it
      been for a some time).
    - In this case, the SOLR tracking attempts to get the parent associations for all nodes,
      regardless of their state of deletion.
    - CLOUD-1807: CloudConv: I get the following error in the solr nodes "Detected stale node entry: NodeVersionKey"
    - MNT-9072: SOLR tracking can be delayed by node deletion 
   51291: JobLockService.releaseLock is not optimistic
    - AcquireLockException is no longer thrown if the lock token is invalid
    - releaseLock returns a boolean to indicate whether the lock was removed or whether it was already gone
    - Prevents unnecessary retrying withing JobLockService implementation (ALF-19098)
   51292: Fix thread safety in PostLookup w.r.t. job lock state; general logging improvements (ALF-19098)
   51293: Fix build after rev 51291 (ALF-19098): JobLockService.releaseLock is now optimistic
   51305: Remainder of fix for CLOUD-1806. RemoteAlfrescoTransformer is unable to serialize transformation options as JSON.
   This check-in just changes Jackson configuration to handle the new TransformationSourceOptions which are now part of TransformationOptions.
   As this field needs to be JSON serialised along with its containing TransformationOptions instance - and particularly because TransformationSourceOptions
   is not a concrete type - we need extra Jackson config in order to correctly marshall and unmarshall the actual concrete types of any instances of TransformationSourceOptions.
   51307: Slight tidy-up of fix for CLOUD-1806.
   I had specified some Jackson config in 3 subclasses and didn't need it as it was already configured on a base type.
   51315: Fix on HEAD-QA only for ALF-19101, which I believe may affect CLOUD-1806.
   A typo in a TransformationSourceOptions method breaks the JavaBeans contract and imperils the JSON serialisation needed for CLOUD-1806.
   I think the method with the typo has been published (in 4.1?) so rather than rename it, I've deprecated it and added a new one with the correct spelling.
   51350: Preparatory work for fixing CLOUD-1754. MailActionExecuterTest failing.
   It has been very difficult to get reliable test feedback on dealing with the MailActionExecuter.
   This is largely because of the difference in the way mail actions are handled in cloud and on enterprise.
   Therefore I have refactored and extended our test code to support fixing the MailActionExecuter.
   Part 1: Refactor the JUnit Rules AlfrescoPeople, AlfrescoPerson, AlfrescoTenant so that when a test is run in
   Community, Enterprise or Cloud the test 'does the right thing' when creating/tearing down test users.
   i.e. On Community/Enterprise, create the user/auth and the person on the single tenant.
   In Cloud create the user in system tenant, create the person in their home tenant, precreating the tenant if necessary
   and tear the whole thing down again.
   I achieved this by having the cm:person-related rules delegate to a spring bean responsible for dealing with test users.
   Then I've got a mechanism to let Enterprise override Community and Cloud override Enterprise.
   So the Community 'TestUserComponent' just creates/deletes users. Enterprise 'inherits' this behaviour.
   The Cloud TestUserComponent deals with the tenant sideshow.
   It's all started in the global-integration-test-context.xml.
   51351: Adding the JIRA number for BDE-173 into this file.
   51353: Fixing some test fallout from pervious commit (51350) which was part of preparatory work for fixing CLOUD-1754.
   Looks like the SyncAudit tests were using ApplicationContextHelper's getApplicationContext method having already started a custom app context?
   51371: Merged BRANCHES/DEV/HEAD42-MERGE to BRANCHES/DEV/HEAD-QA
      51277: Fix for ALF-18948 - Share Hazelcast cluster configuration change required in 4.2
      51313: Minor css improvements to search box
      51367: Fixed order of button styles for lightTheme overrides
      51368: Fix to broken css file - erroneous end comment marker
      51370: Fix for CLOUD-1795 - Login screen isn't resizing to display window
   51406: CLOUD-1573: CloudConv - regression:Preview is not displayed for locked document
      - Added check for presence of container element before attempting setup of document versions data table
   51407: Fix for CLOUD-1754. MailActionExecuterTest test failing.
   The code in MailActionExecuter needs some refactoring. As it was, validation of the mail recipients (are they usernames? email addresses? do the users exist? etc) are not the same for the to-many parameter as they are for the 'to' parameter. There are various other inconsistencies in the code also. I have tried to fix the cases where the inconsistencies are obvious and I've tried to imporove the test coverage too. This code has now become a little too complex and a future JIRA will lead to simplification. Currently the priority is to complete code convergence.
   51410: Fix for     CLOUD-1641   CloudConv - Regression: User from other domain cannot be found
   - cloud queries for people use all a users domains (OK until we partition by tenant)
   Fix for     CLOUD-1768  CloudConv - Regression - CLONE: It's impossible to find the user by any property except the username
   51422: Partial revert of r51410 where accountService was apparently added to the wrong bean definition
   51424: Correct wire up for R 51410 related to
   Fix for     CLOUD-1641   CloudConv - Regression: User from other domain cannot be found
   - cloud queries for people use all a users domains (OK until we partition by tenant)
   Fix for     CLOUD-1768  CloudConv - Regression - CLONE: It's impossible to find the user by any property except the username
   51429: CLOUD-1641: User from other domain cannot be found
   Removing previous fix for CLOUD-1641, the back-end should decide the best route to go depending on the parameters provided. Forcing the query to use a canned query has caused CLOUD-1768.
   51444: Fix for     CLOUD-1641   CloudConv - Regression: User from other domain cannot be found
   Fix for     CLOUD-1768  CloudConv - Regression - CLONE: It's impossible to find the user by any property
   - ignore tenant filter for cloud people query and rely on post query visibility filtering (to see people from other tenants invited in)
   51466: Fix for     CLOUD-1641   CloudConv - Regression: User from other domain cannot be found
   Fix for     CLOUD-1768  CloudConv - Regression - CLONE: It's impossible to find the user by any property
   - exclude permission evaluation when searching for people in the cloud (covered by the visibility stuff)
   51553: CLOUD-1780: CloudConv :- follow button is visible for external user as well
      - Refactored customizations for new dependency loading to fix order of loading
      - Refactored widget userHomeTenant option addition for new instantiation pattern
      - Refactored override of widget's _renderFollowingActions method with new CloudPeopleFinder object that extends PeopleFinder
   51571: Merged HEAD-BUG-FIX to HEAD-QA
      51473: Moved CacheTest back into Repository
      Note for Cluster features: This change no longer relies on any distributed caches and therefore
                                 the new files should just stay as they are and the old ones removed.
   51583: Merged HEAD-BUG-FIX to HEAD-QA
      51477 Merged V4.1-BUG-FIX to HEAD-BUG-FIX
         50995: Support for locking and unlocking of values in the TransactionalCache
                - Infrastructure for MNT-8997: Support definitive writes for TransactionalCache
                - Allows a definitive write (put or remove) call to be made to the TransactionalCache
   51584: Merged HEAD-BUG-FIX to HEAD-QA
      51478: Merged V4.1-BUG-FIX to HEAD-BUG-FIX
         50997: Make use of TransactionalCache's locking ability
                - MNT-8997: Support definitive writes for TransactionalCache
                - PersonServiceImpl and AuthorityDaoImpl both do cache modificiations in 'beforeXYZ' callbacks,
                  which leaves room for further attempted cache updates by other callbacks.
                  Reapplied the fixes using TransactionalCache locks
         51327: Fix  CLOUD-1698 Intermittent invite test failures
                - Uses MNT-8997: Support transactional locking for Authentication Cache removal
   51635: CLOUD-1812 Fix invalid license issue: modify version.properties to have version.label=Cloud and version.edition=Enterprise
   51739: Merged HEAD-BUG-FIX (4.2) to HEAD-QA
      << Low risk change to avoid NPE found testing remote alfresco transformation nodes in DP env >>
      51738: Merged V4.1-BUG-FIX (4.1.6) to HEAD-BUG-FIX (4.2)
         51731: Avoid NPE in ContentServiceImpl.debugActiveTransformers(). Relates to MNT-7158 Investigating adding priorities to transformers
   51752: CLOUD-1812 Fix version.label to simply add a '-Cloud' suffix.
   51753: CLOUD-1822 Change RPM installation folder from tomcat6 to tomcat7
   51760: CLOUD-1385: "Sending a PUT request without a Content-Type header resets the contents mimetype to application/octet-stream" - CMIS was not correctly parsing the mime type header
   51779: Depend on Spring Surf 1.2.0-M6 (r1265) rather than a SNAPSHOT version
   51780: Depend on Spring Surf 1.2.0-M6 (r1265) rather than a SNAPSHOT version
   51783: Merged HEAD to BRANCHES/DEV/HEAD-QA (RECORD ONLY)
      49055:
         ALF-15191: Saving files (new or existing) to Alfresco via CIFS using iWork Pages or Numbers results in invisible files
         Modify HiddenAspect to hide node with explicit='false' by default.
      49062:
         ALF-15191: Saving files (new or existing) to Alfresco via CIFS using iWork Pages or Numbers results in invisible files
         Revert changes that were mistakenly committed.
         Rolled back to r49054
      49075:
         Merged BRANCHES/DEV/HEAD-QA to HEAD:
            49074: ALF-18573: exclude module-context.xml from alfresco-spp-config.jar
      49245:
         Merged HEAD-QA to HEAD (RECORD ONLY)
            49244: Switch version of mvninstall to 4.2.0-QA-SNAPSHOT
      49442:
         CLOUD-1541: "CloudConv: REGRESS - can't connect to cmisatom on HEAD (works on HEAD-QA)": spurious chemistry library appeared
      49599:
         CLOUD-1541: "CloudConv: REGRESS - can't connect to cmisatom on HEAD (works on HEAD-QA)": spurious chemistry library appeared
      49980:
         Merged BRANCHES/DEV/HEAD-QA to HEAD:
            49458: Merged BRANCHES/DEV/HEAD-BUG-FIX to BRANCHES/DEV/HEAD-QA:
                 49117: Merged BRANCHES/DEV/CONV_V413 to BRANCHES/DEV/HEAD-BUG-FIX:
                      49111: - Revert of r47645
                             - Change of imgpreview thumbnailDefinition to mimetype and size expected for HEAD
                             - Relates to ALF-5051 and MNT-8599 / ALF-18726
                 49118: Restored high resolution placeholder icons for imgpreview, relates to ALF-13984
            49979: ALF-18726: Alfresco could not start with WCMQS applied
                 - Updated RenditionServiceIntegrationTest with new imgpreview values
   51793: Increase size of in-transaction "nodeOwnerCache" to 40K items.
   51819: CLOUD-1385: "Sending a PUT request without a Content-Type header resets the contents mimetype to application/octet-stream" - add further tests not covered by original fix
   51836: TenantUtil now gives a bit more information when exceptions pass through (CLOUD-1685)
   51837: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/HEAD-QA:
      51502: Merged BRANCHES/DEV/FEATURES/CLOUD1-SECURITY to BRANCHES/DEV/CLOUD1:
           50389: Created branch for Cloud1 Security Release
           50395: Merged BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIXSF to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY:
                49713: Branch for SalesForce bug fix merges
                49718: Fix for CLOUD-1590 - Session fixation on Cloud specific login page
                49721: CLOUD-1589 Protect Login-page against XSS Cross-Site Scripting attacks
                50320: CLOUD-1653: Improve password strength
                    * Password length constraint previously contained in client-side files has now been removed and is now loaded from the 'Users' scoped config
                    * Minimum password length is increased to 8 characters from 6 previously
                    * The length requirement has been augmented with restrictions on the content, contained in a <password-content> element
                      - A minimum number of uppercase, lowercase, numeric and special characters can be specified
                      - The system-wide default is zero for these new restrictions (i.e. no restrictions)
                    * Alongside the existing <users> element, <network-users> allows the password length or content policies to be tightened for specific networks
                      - Example config is provided in cloud-config.xml showing showing how to do this
                    * Cleaned up existing validation messages and added a custom message based on the content restrictions, where this is configured
                      - Due to ALF-18975 the message appears with raw HTML visible for now
                    * IMPORTANT NOTE: No checking of submitted password values takes place (as before). Users can by-pass the length and content checks by mimiking the web browser themselves.
                      - It is not feasible to add this as part of this work unit, since there are multiple form targets in use, some of which are in the repository. There are no password policies defined in the repository, so these would need adding.
                50358: CLOUD-1707: Regress: Copyright logo is incorrectly encoded on Login and My Dashboard pages for French locale
           50408: CLOUD-1468: Red bar in Tasks "error loading items".
           50409: Fix for CLOUD-1721
           50413: Merge of rev 50185 - XSS attack hardening - incorrect use of ?js_string for safe HTML element IDs - should be ?html
           50576: Merged alfresco/BRANCHES/DEV/FEATURES/CLOUD1-BUG-FIXSF to alfresco/BRANCHES/DEV/FEATURES/CLOUD1-SECURITY:
                50518: ALF-19012: Form validation messages shown in Bubble widgets are not consistent with the input title attribute value
                50574: ALF-18975: HTML in forms validation messages appears as raw HTML in the UI
                    * HTML messages added via Alfresco.forms.Form.addValidation() method no longer escape HTML entities in validation messages presented to the user
                    * The 'message' parameter in addValidation() can now be an object with separate 'html' and 'text' properties. Not all methods of displaying messages to the user support HTML.
                      - If no separate HTML and text values are supplied, the message is assumed to be HTML and is converted to text by removing HTML elements via a regexp
                50575: CLOUD-1653: Supply separate HTML and text validation messages for password content validation failures, since HTML cannot always be displayed
           50577: CLOUD-1653: Use separate HTML and text validation messages for Reset Password and Account Completion pages
           50731: CLOUD-1722: Apply config for Aetna specific password policy.
           50745: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY:
                50622: Fix for     CLOUD-1751   Users searching in their home tenant (or other person related operations) should not be subject to visibility checks
                50664: Fix for     CLOUD-1751   Users searching in their home tenant (or other person related operations) should not be subject to visibility checks
           50748: Merged BRANCHES/DEV/CLOUD1 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY:
                50568: (RECORD ONLY) Merged BRANCHES/DEV/FEATURES/CLOUD1-SECURITY to BRANCHES/DEV/CLOUD1
                      50423: Merged PATCHES/V4.1.1 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY
                                44129: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                                44130: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                                44129: MNT-223: Unbound SOLR result set (from Explorer trashcan query) consumes heap
                             Merged PATCHES/V4.1.4 to BRANCHES/DEV/FEATURES/CLOUD1-SECURITY
                                45951: Fix for ALF-17687 BM-0013: Soak: Run 02: SolrJSONResultSet must preload nodes
                50624: (RECORD ONLY) Delete generation of installers in continuous, to allow 'amber' builds
           50869: CLOUD-1773: Cloud1Security branch build8. User cannot create site
           50872: CLOUD-1653: Ensure password policy is always loaded based on home network, rather than the current context, which could be different.
           51156: CLOUD-1653: Add global flag to regexps to ensure that the number of digits and symbols is correctly counted
           51175: CLOUD-1682: "Hidden File Rename is Broken": fix and make sure the hidden aspect tests run as part of the build
           51183: CLOUD-1682: "Hidden File Rename is Broken": fix up test suite
           51187: CLOUD-1682: "Hidden File Rename is Broken": try again
           51349: CLOUD-1682: "Hidden File Rename is Broken": further tests, deal with hidden aspect manipulation by clients during rename
           51383: CLOUD-1682: "Hidden File Rename is Broken": deal properly with all client controlled hidden aspect cases
   51843: CLOUD-1815: OPTIONS request for force.com not returning 200
   Removed comment around publicapi URL pattern for CORS filter.
   51845: Fixed ALF-19128: Artifact alfresco-repository-*-config.jar contains test files and configuration
    - Move test files into test-resources
   51853: ALF-9096 - CLOUD-1825: Fixed issue with ordering of TransactionSynchronizationAdapter when Activiti job fails
   51859: Merged PATCHES/V4.1.4 to HEAD-QA (REMERGE)
      47738: ALF-18301 (now MNT-6342): ... NPE ... when debug logging for '...AbstractAsynchronouslyRefreshedCache' 
      Merge note: The fix was not present even though it went to HEAD on 8 March (rev 47807)
   51889: Upgrade Spring Surf to 1.2.0-M7 (r1274)
   51893: Removed old Activiti libraries that were conflicting with new ones and causing compilation failure.
   51895: Merge HEAD-BUG-FIX to HEAD-QA (4.2)
      50984: BDE-103: Switch joda-time dependency to core
   51896: Upgrade activiti to 5.11-alf-20130627 in POM for Cloud
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@51909 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			2798 lines
		
	
	
		
			104 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			2798 lines
		
	
	
		
			104 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2013 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.opencmis;
 | |
| 
 | |
| import java.io.BufferedInputStream;
 | |
| import java.io.File;
 | |
| import java.io.FileInputStream;
 | |
| import java.io.InputStream;
 | |
| import java.io.Serializable;
 | |
| import java.math.BigInteger;
 | |
| import java.nio.charset.Charset;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Arrays;
 | |
| import java.util.Collections;
 | |
| import java.util.HashMap;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| 
 | |
| import javax.servlet.http.HttpServletRequest;
 | |
| 
 | |
| import net.sf.acegisecurity.Authentication;
 | |
| 
 | |
| import org.alfresco.cmis.CMISDictionaryModel;
 | |
| import org.alfresco.cmis.CMISInvalidArgumentException;
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.opencmis.dictionary.CMISNodeInfo;
 | |
| import org.alfresco.opencmis.dictionary.CMISObjectVariant;
 | |
| import org.alfresco.opencmis.dictionary.FolderTypeDefintionWrapper;
 | |
| import org.alfresco.opencmis.dictionary.PropertyDefinitionWrapper;
 | |
| import org.alfresco.opencmis.dictionary.TypeDefinitionWrapper;
 | |
| import org.alfresco.query.PagingRequest;
 | |
| import org.alfresco.query.PagingResults;
 | |
| import org.alfresco.repo.content.encoding.ContentCharsetFinder;
 | |
| import org.alfresco.repo.node.getchildren.GetChildrenCannedQuery;
 | |
| import org.alfresco.repo.search.QueryParameterDefImpl;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil;
 | |
| import org.alfresco.repo.security.authentication.Authorization;
 | |
| import org.alfresco.repo.version.VersionModel;
 | |
| import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
 | |
| 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.ContentWriter;
 | |
| import org.alfresco.service.cmr.repository.InvalidNodeRefException;
 | |
| import org.alfresco.service.cmr.repository.NodeRef;
 | |
| import org.alfresco.service.cmr.search.QueryParameterDefinition;
 | |
| import org.alfresco.service.cmr.search.ResultSet;
 | |
| import org.alfresco.service.cmr.search.SearchParameters;
 | |
| import org.alfresco.service.cmr.search.SearchService;
 | |
| 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.namespace.NamespaceService;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.service.namespace.RegexQNamePattern;
 | |
| import org.alfresco.util.Pair;
 | |
| import org.alfresco.util.TempFileProvider;
 | |
| import org.apache.chemistry.opencmis.commons.PropertyIds;
 | |
| import org.apache.chemistry.opencmis.commons.data.Acl;
 | |
| import org.apache.chemistry.opencmis.commons.data.AllowableActions;
 | |
| import org.apache.chemistry.opencmis.commons.data.ContentStream;
 | |
| import org.apache.chemistry.opencmis.commons.data.ExtensionsData;
 | |
| import org.apache.chemistry.opencmis.commons.data.FailedToDeleteData;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectData;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectInFolderContainer;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectInFolderData;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectInFolderList;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectList;
 | |
| import org.apache.chemistry.opencmis.commons.data.ObjectParentData;
 | |
| import org.apache.chemistry.opencmis.commons.data.Properties;
 | |
| import org.apache.chemistry.opencmis.commons.data.RenditionData;
 | |
| import org.apache.chemistry.opencmis.commons.data.RepositoryInfo;
 | |
| import org.apache.chemistry.opencmis.commons.definitions.DocumentTypeDefinition;
 | |
| import org.apache.chemistry.opencmis.commons.definitions.TypeDefinition;
 | |
| import org.apache.chemistry.opencmis.commons.definitions.TypeDefinitionContainer;
 | |
| import org.apache.chemistry.opencmis.commons.definitions.TypeDefinitionList;
 | |
| import org.apache.chemistry.opencmis.commons.enums.AclPropagation;
 | |
| import org.apache.chemistry.opencmis.commons.enums.BaseTypeId;
 | |
| import org.apache.chemistry.opencmis.commons.enums.ContentStreamAllowed;
 | |
| import org.apache.chemistry.opencmis.commons.enums.IncludeRelationships;
 | |
| import org.apache.chemistry.opencmis.commons.enums.RelationshipDirection;
 | |
| import org.apache.chemistry.opencmis.commons.enums.UnfileObject;
 | |
| import org.apache.chemistry.opencmis.commons.enums.VersioningState;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisConstraintException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisContentAlreadyExistsException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisInvalidArgumentException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisObjectNotFoundException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisRuntimeException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisStorageException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisStreamNotSupportedException;
 | |
| import org.apache.chemistry.opencmis.commons.exceptions.CmisVersioningException;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.AccessControlListImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.FailedToDeleteDataImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.ObjectInFolderContainerImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.ObjectInFolderDataImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.ObjectInFolderListImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.ObjectListImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.ObjectParentDataImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.TypeDefinitionContainerImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.dataobjects.TypeDefinitionListImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.server.AbstractCmisService;
 | |
| import org.apache.chemistry.opencmis.commons.impl.server.ObjectInfoImpl;
 | |
| import org.apache.chemistry.opencmis.commons.impl.server.RenditionInfoImpl;
 | |
| import org.apache.chemistry.opencmis.commons.server.CallContext;
 | |
| import org.apache.chemistry.opencmis.commons.server.ObjectInfo;
 | |
| import org.apache.chemistry.opencmis.commons.server.RenditionInfo;
 | |
| import org.apache.chemistry.opencmis.commons.spi.Holder;
 | |
| import org.apache.commons.logging.Log;
 | |
| import org.apache.commons.logging.LogFactory;
 | |
| 
 | |
| /**
 | |
|  * OpenCMIS service implementation
 | |
|  * 
 | |
|  * @author florian.mueller
 | |
|  * @author Derek Hulley
 | |
|  * @since 4.0
 | |
|  */
 | |
| public class AlfrescoCmisServiceImpl extends AbstractCmisService implements AlfrescoCmisService
 | |
| {
 | |
|     private static Log logger = LogFactory.getLog(AlfrescoCmisService.class);
 | |
| 
 | |
|     private static final QName PARAM_PARENT = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, "parent");
 | |
|     private static final QName PARAM_USERNAME = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, "username");
 | |
|     private static final String LUCENE_QUERY_CHECKEDOUT = "+@cm\\:workingCopyOwner:${cm:username}";
 | |
|     private static final String LUCENE_QUERY_CHECKEDOUT_IN_FOLDER = "+@cm\\:workingCopyOwner:${cm:username} +PARENT:\"${cm:parent}\"";
 | |
| 
 | |
|     private static final String MIN_FILTER = "cmis:name,cmis:baseTypeId,cmis:objectTypeId,"
 | |
|             + "cmis:createdBy,cmis:creationDate,cmis:lastModifiedBy,cmis:lastModificationDate,"
 | |
|             + "cmis:contentStreamLength,cmis:contentStreamMimeType,cmis:contentStreamFileName,"
 | |
|             + "cmis:contentStreamId";
 | |
| 
 | |
|     private CMISConnector connector;
 | |
|     private CallContext context;
 | |
|     private Authentication authentication;
 | |
|     private Map<String, CMISNodeInfo> nodeInfoMap;
 | |
|     private Map<String, ObjectInfo> objectInfoMap;
 | |
| 
 | |
|     public AlfrescoCmisServiceImpl(CMISConnector connector)
 | |
|     {
 | |
|         this.connector = connector;
 | |
|         nodeInfoMap = new HashMap<String, CMISNodeInfo>();
 | |
|         objectInfoMap = new HashMap<String, ObjectInfo>();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void open(CallContext context)
 | |
|     {
 | |
|         this.context = context;
 | |
|     }
 | |
|     
 | |
|     protected CallContext getContext()
 | |
|     {
 | |
|         return context;
 | |
|     }
 | |
|     
 | |
|     @Override
 | |
|     public void close()
 | |
|     {
 | |
|         // Put these resources on the transactions
 | |
|         context = null;
 | |
|         nodeInfoMap.clear();
 | |
|         objectInfoMap.clear();
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfoImpl createNodeInfo(NodeRef nodeRef)
 | |
|     {
 | |
|         return createNodeInfo(nodeRef, null);
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfoImpl createNodeInfo(NodeRef nodeRef, VersionHistory versionHistory)
 | |
|     {
 | |
|         CMISNodeInfoImpl result = connector.createNodeInfo(nodeRef, versionHistory);
 | |
|         nodeInfoMap.put(result.getObjectId(), result);
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
|     
 | |
|     protected CMISNodeInfo createNodeInfo(AssociationRef assocRef)
 | |
|     {
 | |
|         CMISNodeInfoImpl result = connector.createNodeInfo(assocRef);
 | |
|         nodeInfoMap.put(result.getObjectId(), result);
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfo getOrCreateNodeInfo(String objectId)
 | |
|     {
 | |
|         CMISNodeInfo result = nodeInfoMap.get(objectId);
 | |
|         if (result == null)
 | |
|         {
 | |
|             result = connector.createNodeInfo(objectId);
 | |
|             nodeInfoMap.put(objectId, result);
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfo getOrCreateNodeInfo(String objectId, String what)
 | |
|     {
 | |
|         CMISNodeInfo result = getOrCreateNodeInfo(objectId);
 | |
|         if (result instanceof CMISNodeInfoImpl)
 | |
|         {
 | |
|             ((CMISNodeInfoImpl) result).checkIfUseful(what);
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfo getOrCreateFolderInfo(String folderId, String what)
 | |
|     {
 | |
|         CMISNodeInfo result = getOrCreateNodeInfo(folderId);
 | |
|         if (result instanceof CMISNodeInfoImpl)
 | |
|         {
 | |
|             ((CMISNodeInfoImpl) result).checkIfFolder(what);
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     protected CMISNodeInfo addNodeInfo(CMISNodeInfo info)
 | |
|     {
 | |
|         nodeInfoMap.put(info.getObjectId(), info);
 | |
| 
 | |
|         return info;
 | |
|     }
 | |
| 
 | |
|     // --- repository service ---
 | |
| 
 | |
|     @Override
 | |
|     public List<RepositoryInfo> getRepositoryInfos(ExtensionsData extension)
 | |
|     {
 | |
|         return Collections.singletonList(connector.getRepositoryInfo());
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public RepositoryInfo getRepositoryInfo(String repositoryId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         return connector.getRepositoryInfo();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public TypeDefinitionList getTypeChildren(
 | |
|             String repositoryId, String typeId, Boolean includePropertyDefinitions,
 | |
|             BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // convert BigIntegers to int
 | |
|         int max = (maxItems == null ? Integer.MAX_VALUE : maxItems.intValue());
 | |
|         int skip = (skipCount == null || skipCount.intValue() < 0 ? 0 : skipCount.intValue());
 | |
| 
 | |
|         // set up the result
 | |
|         TypeDefinitionListImpl result = new TypeDefinitionListImpl();
 | |
|         List<TypeDefinition> list = new ArrayList<TypeDefinition>();
 | |
|         result.setList(list);
 | |
| 
 | |
|         // get the types from the dictionary
 | |
|         List<TypeDefinitionWrapper> childrenList;
 | |
|         if (typeId == null)
 | |
|         {
 | |
|             childrenList = connector.getOpenCMISDictionaryService().getBaseTypes();
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             TypeDefinitionWrapper tdw = connector.getOpenCMISDictionaryService().findType(typeId);
 | |
|             if (tdw == null)
 | |
|             {
 | |
|                 throw new CmisObjectNotFoundException("Type '" + typeId + "' is unknown!");
 | |
|             }
 | |
|             childrenList = tdw.getChildren();
 | |
|         }
 | |
| 
 | |
|         // create result
 | |
|         if (max > 0)
 | |
|         {
 | |
|             int lastIndex = (max + skip > childrenList.size() ? childrenList.size() : max + skip) - 1;
 | |
|             for (int i = skip; i <= lastIndex; i++)
 | |
|             {
 | |
|                 list.add(childrenList.get(i).getTypeDefinition(includePropertyDefinitions));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         result.setHasMoreItems(childrenList.size() - skip > result.getList().size());
 | |
|         result.setNumItems(BigInteger.valueOf(childrenList.size()));
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public TypeDefinition getTypeDefinition(String repositoryId, String typeId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // find the type
 | |
|         TypeDefinitionWrapper tdw = connector.getOpenCMISDictionaryService().findType(typeId);
 | |
|         if (tdw == null)
 | |
|         {
 | |
|             throw new CmisObjectNotFoundException("Type '" + typeId + "' is unknown!");
 | |
|         }
 | |
| 
 | |
|         // return type definition
 | |
|         return tdw.getTypeDefinition(true);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<TypeDefinitionContainer> getTypeDescendants(
 | |
|             String repositoryId, String typeId, BigInteger depth,
 | |
|             Boolean includePropertyDefinitions, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         List<TypeDefinitionContainer> result = new ArrayList<TypeDefinitionContainer>();
 | |
| 
 | |
|         // check depth
 | |
|         int d = (depth == null ? -1 : depth.intValue());
 | |
|         if (d == 0)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Depth must not be 0!");
 | |
|         }
 | |
| 
 | |
|         if (typeId == null)
 | |
|         {
 | |
|             for (TypeDefinitionWrapper tdw : connector.getOpenCMISDictionaryService().getBaseTypes())
 | |
|             {
 | |
|                 result.add(getTypesDescendants(d, tdw, includePropertyDefinitions));
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             TypeDefinitionWrapper tdw = connector.getOpenCMISDictionaryService().findType(typeId);
 | |
|             if (tdw == null)
 | |
|             {
 | |
|                 throw new CmisObjectNotFoundException("Type '" + typeId + "' is unknown!");
 | |
|             }
 | |
| 
 | |
|             if (tdw.getChildren() != null)
 | |
|             {
 | |
|                 for (TypeDefinitionWrapper child : tdw.getChildren())
 | |
|                 {
 | |
|                     result.add(getTypesDescendants(d, child, includePropertyDefinitions));
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Gathers the type descendants tree.
 | |
|      */
 | |
|     private TypeDefinitionContainer getTypesDescendants(
 | |
|             int depth, TypeDefinitionWrapper tdw, boolean includePropertyDefinitions)
 | |
|     {
 | |
|         TypeDefinitionContainerImpl result = new TypeDefinitionContainerImpl();
 | |
| 
 | |
|         result.setTypeDefinition(tdw.getTypeDefinition(includePropertyDefinitions));
 | |
| 
 | |
|         if (depth != 0)
 | |
|         {
 | |
|             if (tdw.getChildren() != null)
 | |
|             {
 | |
|                 result.setChildren(new ArrayList<TypeDefinitionContainer>());
 | |
|                 for (TypeDefinitionWrapper tdc : tdw.getChildren())
 | |
|                 {
 | |
|                     result.getChildren().add(
 | |
|                             getTypesDescendants(depth < 0 ? -1 : depth - 1, tdc, includePropertyDefinitions));
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     // --- navigation service ---
 | |
| 
 | |
|     @Override
 | |
|     public ObjectInFolderList getChildren(
 | |
|             String repositoryId, String folderId, String filter, String orderBy,
 | |
|             Boolean includeAllowableActions, IncludeRelationships includeRelationships, String renditionFilter,
 | |
|             Boolean includePathSegment, BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         long start = System.currentTimeMillis();
 | |
| 
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // convert BigIntegers to int
 | |
|         int max = (maxItems == null ? Integer.MAX_VALUE : maxItems.intValue());
 | |
|         int skip = (skipCount == null || skipCount.intValue() < 0 ? 0 : skipCount.intValue());
 | |
| 
 | |
|         ObjectInFolderListImpl result = new ObjectInFolderListImpl();
 | |
|         List<ObjectInFolderData> list = new ArrayList<ObjectInFolderData>();
 | |
|         result.setObjects(list);
 | |
| 
 | |
|         // get the children references
 | |
|         NodeRef folderNodeRef = getOrCreateFolderInfo(folderId, "Folder").getNodeRef();
 | |
| 
 | |
|         // convert orderBy to sortProps
 | |
|         List<Pair<QName, Boolean>> sortProps = null;
 | |
|         if (orderBy != null)
 | |
|         {
 | |
|             sortProps = new ArrayList<Pair<QName, Boolean>>(1);
 | |
| 
 | |
|             String[] parts = orderBy.split(",");
 | |
|             int len = parts.length;
 | |
|             final int origLen = len;
 | |
| 
 | |
|             if (origLen > 0)
 | |
|             {
 | |
|                 int maxSortProps = GetChildrenCannedQuery.MAX_FILTER_SORT_PROPS;
 | |
|                 if (len > maxSortProps)
 | |
|                 {
 | |
|                     if (logger.isDebugEnabled())
 | |
|                     {
 | |
|                         logger.debug(
 | |
|                                 "Too many sort properties in 'orderBy' - ignore those above max (max="
 | |
|                                 + maxSortProps + ",actual=" + len + ")");
 | |
|                     }
 | |
|                     len = maxSortProps;
 | |
|                 }
 | |
|                 for (int i = 0; i < len; i++)
 | |
|                 {
 | |
|                     String[] sort = parts[i].split(" +");
 | |
|                     
 | |
|                     if (sort.length > 0)
 | |
|                     {
 | |
|                         PropertyDefinitionWrapper propDef = connector.getOpenCMISDictionaryService()
 | |
|                                 .findPropertyByQueryName(sort[0]);
 | |
|                         if (propDef != null)
 | |
|                         {
 | |
|                             QName sortProp = null;
 | |
|                             if (propDef.getPropertyId().equals(CMISDictionaryModel.PROP_BASE_TYPE_ID))
 | |
|                             {
 | |
|                                 // special-case (see also ALF-13968) - for getChildren, using "cmis:baseTypeId" allows sorting of folders first and vice-versa (cmis:folder <-> cmis:document)
 | |
|                                 sortProp = GetChildrenCannedQuery.SORT_QNAME_NODE_IS_FOLDER;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 sortProp = propDef.getPropertyAccessor().getMappedProperty();
 | |
|                             }
 | |
|                             
 | |
|                             if (sortProp != null)
 | |
|                             {
 | |
|                                 boolean sortAsc = (sort.length == 1) || sort[1].equalsIgnoreCase("asc");
 | |
|                                 sortProps.add(new Pair<QName, Boolean>(sortProp, sortAsc));
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 if (logger.isDebugEnabled())
 | |
|                                 {
 | |
|                                     logger.debug("Ignore sort property '" + sort[0] + " - mapping not found");
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             if (logger.isDebugEnabled())
 | |
|                             {
 | |
|                                 logger.debug("Ignore sort property '" + sort[0] + " - query name not found");
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (sortProps.size() < origLen)
 | |
|             {
 | |
|                 logger.warn("Sort properties trimmed - either too many and/or not found: \n" + "   orig:  " + orderBy
 | |
|                         + "\n" + "   final: " + sortProps);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         PagingRequest pageRequest = new PagingRequest(skip, max, null);
 | |
|         pageRequest.setRequestTotalCountMax(skip + 10000); // TODO make this optional/configurable
 | |
|                                                            // - affects whether numItems may be returned
 | |
| 
 | |
|         PagingResults<FileInfo> pageOfNodeInfos = connector.getFileFolderService().list(
 | |
|                 folderNodeRef, true, true,
 | |
|                 null, sortProps, pageRequest);
 | |
| 
 | |
|         if (max > 0)
 | |
|         {
 | |
|             for (FileInfo child : pageOfNodeInfos.getPage())
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     if(connector.filter(child.getNodeRef()))
 | |
|                     {
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     // create a child CMIS object
 | |
|                     CMISNodeInfo ni = createNodeInfo(child.getNodeRef());
 | |
|                     
 | |
|                     if (getObjectInfo(repositoryId, ni.getObjectId())==null)
 | |
|                     {
 | |
|                         // ignore invalid children
 | |
|                         continue;
 | |
|                     }
 | |
|                     
 | |
|                     if (CMISObjectVariant.NOT_A_CMIS_OBJECT.equals(ni.getObjectVariant()))
 | |
|                     {
 | |
|                         continue;  //Skip non-cmis objects
 | |
|                     }
 | |
|                     
 | |
|                     ObjectData object = connector.createCMISObject(ni, child, filter, includeAllowableActions,
 | |
|                             includeRelationships, renditionFilter, false, false);
 | |
| 
 | |
|                     if (getContext().isObjectInfoRequired())
 | |
|                     {
 | |
|                         getObjectInfo(repositoryId, ni.getObjectId(), includeRelationships);
 | |
|                     }
 | |
| 
 | |
|                     ObjectInFolderDataImpl childData = new ObjectInFolderDataImpl();
 | |
|                     childData.setObject(object);
 | |
| 
 | |
|                     // include path segment
 | |
|                     if (includePathSegment)
 | |
|                     {
 | |
|                         childData.setPathSegment(child.getName());
 | |
|                     }
 | |
| 
 | |
|                     // add it
 | |
|                     list.add(childData);
 | |
|                 }
 | |
|                 catch (InvalidNodeRefException e)
 | |
|                 {
 | |
|                     // ignore invalid children
 | |
|                 }
 | |
|                 catch(CmisObjectNotFoundException e)
 | |
|                 {
 | |
|                     // ignore objects that have not been found (perhaps because their type is unknown to CMIS)
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // has more ?
 | |
|         result.setHasMoreItems(pageOfNodeInfos.hasMoreItems());
 | |
| 
 | |
|         // total count ?
 | |
|         Pair<Integer, Integer> totalCounts = pageOfNodeInfos.getTotalResultCount();
 | |
|         if (totalCounts != null)
 | |
|         {
 | |
|             Integer totalCountLower = totalCounts.getFirst();
 | |
|             Integer totalCountUpper = totalCounts.getSecond();
 | |
|             if ((totalCountLower != null) && (totalCountLower.equals(totalCountUpper)))
 | |
|             {
 | |
|                 result.setNumItems(BigInteger.valueOf(totalCountLower));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("getChildren: " + list.size() + " in " + (System.currentTimeMillis() - start) + " msecs");
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<ObjectInFolderContainer> getDescendants(
 | |
|             String repositoryId, String folderId, BigInteger depth,
 | |
|             String filter, Boolean includeAllowableActions, IncludeRelationships includeRelationships,
 | |
|             String renditionFilter, Boolean includePathSegment, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         List<ObjectInFolderContainer> result = new ArrayList<ObjectInFolderContainer>();
 | |
| 
 | |
|         getDescendantsTree(
 | |
|                 repositoryId,
 | |
|                 getOrCreateFolderInfo(folderId, "Folder").getNodeRef(),
 | |
|                 depth.intValue(),
 | |
|                 filter,
 | |
|                 includeAllowableActions, includeRelationships, renditionFilter, includePathSegment, false,
 | |
|                 result);
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<ObjectInFolderContainer> getFolderTree(
 | |
|             String repositoryId, String folderId, BigInteger depth,
 | |
|             String filter, Boolean includeAllowableActions, IncludeRelationships includeRelationships,
 | |
|             String renditionFilter, Boolean includePathSegment, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         List<ObjectInFolderContainer> result = new ArrayList<ObjectInFolderContainer>();
 | |
| 
 | |
|         getDescendantsTree(
 | |
|                 repositoryId,
 | |
|                 getOrCreateFolderInfo(folderId, "Folder").getNodeRef(),
 | |
|                 depth.intValue(),
 | |
|                 filter, includeAllowableActions, includeRelationships, renditionFilter, includePathSegment, true,
 | |
|                 result);
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     private void getDescendantsTree(
 | |
|             String repositoryId, NodeRef folderNodeRef, int depth, String filter,
 | |
|             Boolean includeAllowableActions, IncludeRelationships includeRelationships, String renditionFilter,
 | |
|             Boolean includePathSegment, boolean foldersOnly, List<ObjectInFolderContainer> list)
 | |
|     {
 | |
|         // get the children references
 | |
|         List<ChildAssociationRef> childrenList = connector.getNodeService().getChildAssocs(folderNodeRef);
 | |
|         for (ChildAssociationRef child : childrenList)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 TypeDefinitionWrapper type = connector.getType(child.getChildRef());
 | |
|                 if (type == null)
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 boolean isFolder = (type instanceof FolderTypeDefintionWrapper);
 | |
| 
 | |
|                 if (foldersOnly && !isFolder)
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 
 | |
|                 if(isFolder && type.getAlfrescoClass().equals(ContentModel.TYPE_SYSTEM_FOLDER))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 
 | |
|                 if(connector.isHidden(child.getChildRef()))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if(connector.filter(child.getChildRef()))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 
 | |
|                 // create a child CMIS object
 | |
|                 ObjectInFolderDataImpl object = new ObjectInFolderDataImpl();
 | |
|                 CMISNodeInfo ni = createNodeInfo(child.getChildRef());
 | |
|                 object.setObject(connector.createCMISObject(
 | |
|                         ni, filter, includeAllowableActions, includeRelationships,
 | |
|                         renditionFilter, false, false));
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, ni.getObjectId(), includeRelationships);
 | |
|                 }
 | |
| 
 | |
|                 if (includePathSegment)
 | |
|                 {
 | |
|                     object.setPathSegment(connector.getName(child.getChildRef()));
 | |
|                 }
 | |
| 
 | |
|                 // create the container
 | |
|                 ObjectInFolderContainerImpl container = new ObjectInFolderContainerImpl();
 | |
|                 container.setObject(object);
 | |
| 
 | |
|                 if ((depth != 1) && isFolder)
 | |
|                 {
 | |
|                     container.setChildren(new ArrayList<ObjectInFolderContainer>());
 | |
|                     getDescendantsTree(
 | |
|                             repositoryId,
 | |
|                             child.getChildRef(),
 | |
|                             depth - 1, filter, includeAllowableActions,
 | |
|                             includeRelationships, renditionFilter, includePathSegment, foldersOnly,
 | |
|                             container.getChildren());
 | |
|                 }
 | |
| 
 | |
|                 // add it
 | |
|                 list.add(container);
 | |
|             }
 | |
|             catch (InvalidNodeRefException e)
 | |
|             {
 | |
|                 // ignore invalid children
 | |
|             }
 | |
|             catch(CmisObjectNotFoundException e)
 | |
|             {
 | |
|                 // ignore objects that have not been found (perhaps because their type is unknown to CMIS)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectData getFolderParent(String repositoryId, String folderId, String filter, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get the node ref
 | |
|         CMISNodeInfo info = getOrCreateFolderInfo(folderId, "Folder");
 | |
| 
 | |
|         // the root folder has no parent
 | |
|         if (info.isRootFolder())
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Root folder has no parent!");
 | |
|         }
 | |
| 
 | |
|         // get the parent
 | |
|         List<CMISNodeInfo> parentInfos = info.getParents();
 | |
|         if (parentInfos.isEmpty())
 | |
|         {
 | |
|             throw new CmisRuntimeException("Folder has no parent and is not the root folder?!");
 | |
|         }
 | |
| 
 | |
|         CMISNodeInfo parentInfo = addNodeInfo(parentInfos.get(0));
 | |
| 
 | |
|         ObjectData result = connector.createCMISObject(
 | |
|                 parentInfo, filter, false, IncludeRelationships.NONE,
 | |
|                 CMISConnector.RENDITION_NONE, false, false);
 | |
|         if (getContext().isObjectInfoRequired())
 | |
|         {
 | |
|             getObjectInfo(
 | |
|                     repositoryId,
 | |
|                     parentInfo.getObjectId(),
 | |
|                     IncludeRelationships.NONE);
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<ObjectParentData> getObjectParents(
 | |
|             String repositoryId, String objectId, String filter,
 | |
|             Boolean includeAllowableActions, IncludeRelationships includeRelationships, String renditionFilter,
 | |
|             Boolean includeRelativePathSegment, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         List<ObjectParentData> result = new ArrayList<ObjectParentData>();
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // relationships are not filed
 | |
|         if (info.isRelationship())
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not fileable!");
 | |
|         }
 | |
| 
 | |
|         if (info.isFolder() && !info.isRootFolder())
 | |
|         {
 | |
|             List<CMISNodeInfo> parentInfos = info.getParents();
 | |
|             if (!parentInfos.isEmpty())
 | |
|             {
 | |
|                 CMISNodeInfo parentInfo = addNodeInfo(parentInfos.get(0));
 | |
| 
 | |
|                 ObjectData object = connector.createCMISObject(
 | |
|                         parentInfo, filter, includeAllowableActions,
 | |
|                         includeRelationships, renditionFilter, false, false);
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, object.getId(), includeRelationships);
 | |
|                 }
 | |
| 
 | |
|                 ObjectParentDataImpl objectParent = new ObjectParentDataImpl();
 | |
|                 objectParent.setObject(object);
 | |
| 
 | |
|                 // include relative path segment
 | |
|                 if (includeRelativePathSegment)
 | |
|                 {
 | |
|                     objectParent.setRelativePathSegment(info.getName());
 | |
|                 }
 | |
| 
 | |
|                 result.add(objectParent);
 | |
|             }
 | |
|         }
 | |
|         else if (info.isCurrentVersion() || info.isPWC())
 | |
|         {
 | |
|             List<CMISNodeInfo> parentInfos = info.getParents();
 | |
|             for (CMISNodeInfo parentInfo : parentInfos)
 | |
|             {
 | |
|                 addNodeInfo(parentInfo);
 | |
| 
 | |
|                 ObjectData object = connector.createCMISObject(
 | |
|                         parentInfo, filter, includeAllowableActions,
 | |
|                         includeRelationships, renditionFilter, false, false);
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, object.getId(), includeRelationships);
 | |
|                 }
 | |
| 
 | |
|                 ObjectParentDataImpl objectParent = new ObjectParentDataImpl();
 | |
|                 objectParent.setObject(object);
 | |
| 
 | |
|                 // include relative path segment
 | |
|                 if (includeRelativePathSegment)
 | |
|                 {
 | |
|                     objectParent.setRelativePathSegment(info.getName());
 | |
|                 }
 | |
| 
 | |
|                 result.add(objectParent);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectList getCheckedOutDocs(
 | |
|             String repositoryId, String folderId, String filter, String orderBy,
 | |
|             Boolean includeAllowableActions, IncludeRelationships includeRelationships, String renditionFilter,
 | |
|             BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // convert BigIntegers to int
 | |
|         int max = (maxItems == null ? Integer.MAX_VALUE : maxItems.intValue());
 | |
|         int skip = (skipCount == null || skipCount.intValue() < 0 ? 0 : skipCount.intValue());
 | |
| 
 | |
|         // prepare query
 | |
|         SearchParameters params = new SearchParameters();
 | |
|         params.setLanguage(SearchService.LANGUAGE_LUCENE);
 | |
|         QueryParameterDefinition usernameDef = new QueryParameterDefImpl(
 | |
|                 PARAM_USERNAME,
 | |
|                 connector.getDictionaryService().getDataType(DataTypeDefinition.TEXT),
 | |
|                 true,
 | |
|                 AuthenticationUtil.getFullyAuthenticatedUser());
 | |
|         params.addQueryParameterDefinition(usernameDef);
 | |
| 
 | |
|         if (folderId == null)
 | |
|         {
 | |
|             params.setQuery(LUCENE_QUERY_CHECKEDOUT);
 | |
|             params.addStore(connector.getRootStoreRef());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             CMISNodeInfo folderInfo = getOrCreateFolderInfo(folderId, "Folder");
 | |
| 
 | |
|             params.setQuery(LUCENE_QUERY_CHECKEDOUT_IN_FOLDER);
 | |
|             params.addStore(folderInfo.getNodeRef().getStoreRef());
 | |
|             QueryParameterDefinition parentDef = new QueryParameterDefImpl(
 | |
|                     PARAM_PARENT,
 | |
|                     connector.getDictionaryService().getDataType(DataTypeDefinition.NODE_REF),
 | |
|                     true,
 | |
|                     folderInfo.getNodeRef().toString());
 | |
|             params.addQueryParameterDefinition(parentDef);
 | |
|         }
 | |
| 
 | |
|         // set up order
 | |
|         if (orderBy != null)
 | |
|         {
 | |
|             String[] parts = orderBy.split(",");
 | |
|             for (int i = 0; i < parts.length; i++)
 | |
|             {
 | |
|                 String[] sort = parts[i].split(" +");
 | |
| 
 | |
|                 if (sort.length < 1)
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 PropertyDefinitionWrapper propDef = connector.getOpenCMISDictionaryService().findPropertyByQueryName(sort[0]);
 | |
|                 if (propDef != null)
 | |
|                 {
 | |
|                     if (propDef.getPropertyDefinition().isOrderable())
 | |
|                     {
 | |
|                         QName sortProp = propDef.getPropertyAccessor().getMappedProperty();
 | |
|                         if (sortProp != null)
 | |
|                         {
 | |
|                             boolean sortAsc = (sort.length == 1) || sort[1].equalsIgnoreCase("asc");
 | |
|                             params.addSort(propDef.getPropertyLuceneBuilder().getLuceneFieldName(), sortAsc);
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             if (logger.isDebugEnabled())
 | |
|                             {
 | |
|                                 logger.debug("Ignore sort property '" + sort[0] + " - mapping not found");
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         if (logger.isDebugEnabled())
 | |
|                         {
 | |
|                             logger.debug("Ignore sort property '" + sort[0] + " - not orderable");
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if (logger.isDebugEnabled())
 | |
|                     {
 | |
|                         logger.debug("Ignore sort property '" + sort[0] + " - query name not found");
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // execute query
 | |
|         ResultSet resultSet = null;
 | |
|         List<NodeRef> nodeRefs;
 | |
|         try
 | |
|         {
 | |
|             resultSet = connector.getSearchService().query(params);
 | |
|             nodeRefs = resultSet.getNodeRefs();
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             if (resultSet != null)
 | |
|             {
 | |
|                 resultSet.close();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // collect results
 | |
|         ObjectListImpl result = new ObjectListImpl();
 | |
|         List<ObjectData> list = new ArrayList<ObjectData>();
 | |
|         result.setObjects(list);
 | |
| 
 | |
|         int skipCounter = skip;
 | |
|         if (max > 0)
 | |
|         {
 | |
|             for (NodeRef nodeRef : nodeRefs)
 | |
|             {
 | |
|                 // TODO - perhaps filter by path in the query instead?
 | |
|                 if(connector.filter(nodeRef))
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if (skipCounter > 0)
 | |
|                 {
 | |
|                     skipCounter--;
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 if (list.size() == max)
 | |
|                 {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     // create a CMIS object
 | |
|                     CMISNodeInfo ni = createNodeInfo(nodeRef);
 | |
|                     ObjectData object = connector.createCMISObject(
 | |
|                             ni, filter, includeAllowableActions,
 | |
|                             includeRelationships, renditionFilter, false, false);
 | |
| 
 | |
|                     if (getContext().isObjectInfoRequired())
 | |
|                     {
 | |
|                         getObjectInfo(repositoryId, ni.getObjectId(), includeRelationships);
 | |
|                     }
 | |
| 
 | |
|                     // add it
 | |
|                     list.add(object);
 | |
|                 }
 | |
|                 catch (InvalidNodeRefException e)
 | |
|                 {
 | |
|                     // ignore invalid objects
 | |
|                 }
 | |
|                 catch(CmisObjectNotFoundException e)
 | |
|                 {
 | |
|                     // ignore objects that have not been found (perhaps because their type is unknown to CMIS)
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // has more ?
 | |
|         result.setHasMoreItems(nodeRefs.size() - skip > list.size());
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     // --- object service ---
 | |
| 
 | |
|     @Override
 | |
|     public String create(
 | |
|             String repositoryId, Properties properties, String folderId, ContentStream contentStream,
 | |
|             VersioningState versioningState, List<String> policies, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // check properties
 | |
|         if (properties == null || properties.getProperties() == null)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Properties must be set!");
 | |
|         }
 | |
| 
 | |
|         // get the type
 | |
|         String objectTypeId = connector.getObjectTypeIdProperty(properties);
 | |
| 
 | |
|         // find the type
 | |
|         TypeDefinitionWrapper type = connector.getOpenCMISDictionaryService().findType(objectTypeId);
 | |
|         if (type == null)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Type '" + objectTypeId + "' is unknown!");
 | |
|         }
 | |
| 
 | |
|         // create object
 | |
|         String newId = null;
 | |
|         switch (type.getBaseTypeId())
 | |
|         {
 | |
|         case CMIS_DOCUMENT:
 | |
|             newId = createDocument(repositoryId, properties, folderId, contentStream, versioningState, policies, null,
 | |
|                     null, extension);
 | |
|             break;
 | |
|         case CMIS_FOLDER:
 | |
|             newId = createFolder(repositoryId, properties, folderId, policies, null, null, extension);
 | |
|             break;
 | |
|         case CMIS_POLICY:
 | |
|             newId = createPolicy(repositoryId, properties, folderId, policies, null, null, extension);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         // check new object id
 | |
|         if (newId == null)
 | |
|         {
 | |
|             throw new CmisRuntimeException("Creation failed!");
 | |
|         }
 | |
| 
 | |
|         if (getContext().isObjectInfoRequired())
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 getObjectInfo(repositoryId, newId, "*", IncludeRelationships.NONE);
 | |
|             }
 | |
|             catch (InvalidNodeRefException e)
 | |
|             {
 | |
|                 throw new CmisRuntimeException("Creation failed! New object not found!");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // return the new object id
 | |
|         return newId;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public String createFolder(String repositoryId, final Properties properties, String folderId,
 | |
|             final List<String> policies, final Acl addAces, final Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get the parent folder node ref
 | |
|         final CMISNodeInfo parentInfo = getOrCreateFolderInfo(folderId, "Folder");
 | |
| 
 | |
|         // get name and type
 | |
|         final String name = connector.getNameProperty(properties, null);
 | |
|         final String objectTypeId = connector.getObjectTypeIdProperty(properties);
 | |
|         final TypeDefinitionWrapper type = connector.getTypeForCreate(objectTypeId, BaseTypeId.CMIS_FOLDER);
 | |
| 
 | |
|         connector.checkChildObjectType(parentInfo, type.getTypeId());
 | |
| 
 | |
|         // run transaction
 | |
|         FileInfo fileInfo = connector.getFileFolderService().create(
 | |
|                 parentInfo.getNodeRef(), name, type.getAlfrescoClass());
 | |
|         NodeRef nodeRef = fileInfo.getNodeRef();
 | |
| 
 | |
|         connector.setProperties(nodeRef, type, properties, new String[] { PropertyIds.NAME, PropertyIds.OBJECT_TYPE_ID });
 | |
|         connector.applyPolicies(nodeRef, type, policies);
 | |
|         connector.applyACL(nodeRef, type, addAces, removeAces);
 | |
|         
 | |
|         connector.getActivityPoster().postFileFolderAdded(nodeRef);
 | |
| 
 | |
|         return nodeRef.getId();
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public String createDocument(
 | |
|             String repositoryId, final Properties properties, String folderId,
 | |
|             final ContentStream contentStream, final VersioningState versioningState, final List<String> policies,
 | |
|             final Acl addAces, final Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
|         
 | |
|         // get the parent folder node ref
 | |
|         final CMISNodeInfo parentInfo = getOrCreateFolderInfo(folderId, "Parent folder");
 | |
| 
 | |
|         // get name and type
 | |
|         final String name = connector.getNameProperty(properties, null);
 | |
|         final String objectTypeId = connector.getObjectTypeIdProperty(properties);
 | |
|         final TypeDefinitionWrapper type = connector.getTypeForCreate(objectTypeId, BaseTypeId.CMIS_DOCUMENT);
 | |
| 
 | |
|         connector.checkChildObjectType(parentInfo, type.getTypeId());
 | |
| 
 | |
|         DocumentTypeDefinition docType = (DocumentTypeDefinition) type.getTypeDefinition(false);
 | |
| 
 | |
|         if ((docType.getContentStreamAllowed() == ContentStreamAllowed.NOTALLOWED) && (contentStream != null))
 | |
|         {
 | |
|             throw new CmisConstraintException("This document type does not support content!");
 | |
|         }
 | |
| 
 | |
|         if ((docType.getContentStreamAllowed() == ContentStreamAllowed.REQUIRED) && (contentStream == null))
 | |
|         {
 | |
|             throw new CmisConstraintException("This document type does requires content!");
 | |
|         }
 | |
| 
 | |
|         if (docType.isVersionable() && (versioningState == VersioningState.NONE))
 | |
|         {
 | |
|             throw new CmisConstraintException("This document type is versionable!");
 | |
|         }
 | |
| 
 | |
|         if (!docType.isVersionable() && (versioningState != VersioningState.NONE))
 | |
|         {
 | |
|             throw new CmisConstraintException("This document type is not versionable!");
 | |
|         }
 | |
| 
 | |
|         FileInfo fileInfo = connector.getFileFolderService().create(
 | |
|                 parentInfo.getNodeRef(), name, type.getAlfrescoClass());
 | |
|         NodeRef nodeRef = fileInfo.getNodeRef();
 | |
| 
 | |
|         connector.setProperties(nodeRef, type, properties, new String[] { PropertyIds.NAME, PropertyIds.OBJECT_TYPE_ID });
 | |
|         connector.applyPolicies(nodeRef, type, policies);
 | |
|         connector.applyACL(nodeRef, type, addAces, removeAces);
 | |
| 
 | |
|         // handle content
 | |
|         File tempFile = null;
 | |
|         try
 | |
|         {
 | |
| 	        if (contentStream != null)
 | |
| 	        {
 | |
| 	            // write content
 | |
| 	            String mimeType = parseMimeType(contentStream);
 | |
| 
 | |
| 	            // copy stream to temp file
 | |
| 	            // OpenCMIS does this for us ....
 | |
| 	            tempFile = copyToTempFile(contentStream);
 | |
| 	            final Charset encoding = (tempFile == null ? null : getEncoding(tempFile, contentStream.getMimeType()));
 | |
| 	                
 | |
| 	            ContentWriter writer = connector.getFileFolderService().getWriter(nodeRef);
 | |
| 	            writer.setMimetype(mimeType);
 | |
| 	            writer.setEncoding(encoding.name());
 | |
| 	            writer.putContent(tempFile);
 | |
| 	        }
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|         	if(tempFile != null)
 | |
|         	{
 | |
|         		removeTempFile(tempFile);
 | |
|         	}
 | |
|         }
 | |
| 
 | |
|         connector.extractMetadata(nodeRef);
 | |
| 
 | |
|         // generate "doclib" thumbnail asynchronously
 | |
|         connector.createThumbnails(nodeRef, Collections.singleton("doclib"));
 | |
| 
 | |
|         connector.applyVersioningState(nodeRef, versioningState);
 | |
| 
 | |
|         String objectId = connector.createObjectId(nodeRef);
 | |
| 
 | |
|         connector.getActivityPoster().postFileFolderAdded(nodeRef);
 | |
| 
 | |
|         return objectId;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public String createDocumentFromSource(
 | |
|             String repositoryId, String sourceId, final Properties properties,
 | |
|             String folderId, final VersioningState versioningState, final List<String> policies, final Acl addAces,
 | |
|             final Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get the parent folder node ref
 | |
|         final CMISNodeInfo parentInfo = getOrCreateFolderInfo(folderId, "Parent folder");
 | |
| 
 | |
|         // get source
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(sourceId, "Source");
 | |
| 
 | |
|         // check source
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisConstraintException("Source object is not a document!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef sourceNodeRef = info.getNodeRef();
 | |
|         if (!info.isDocument())
 | |
|         {
 | |
|             throw new CmisConstraintException("Source object is not a document!");
 | |
|         }
 | |
| 
 | |
|         // get name and type
 | |
|         final String name = connector.getNameProperty(properties, info.getName());
 | |
| 
 | |
|         final TypeDefinitionWrapper type = info.getType();
 | |
|         connector.checkChildObjectType(parentInfo, type.getTypeId());
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             FileInfo fileInfo = connector.getFileFolderService().copy(
 | |
|                     sourceNodeRef, parentInfo.getNodeRef(), name);
 | |
|             NodeRef nodeRef = fileInfo.getNodeRef();
 | |
|             connector.setProperties(nodeRef, type, properties, new String[] {
 | |
|                     PropertyIds.NAME, PropertyIds.OBJECT_TYPE_ID });
 | |
|             connector.applyPolicies(nodeRef, type, policies);
 | |
|             connector.applyACL(nodeRef, type, addAces, removeAces);
 | |
|             
 | |
|             connector.extractMetadata(nodeRef);
 | |
|             connector.createThumbnails(nodeRef, Collections.singleton("doclib"));
 | |
| 
 | |
|             connector.applyVersioningState(nodeRef, versioningState);
 | |
| 
 | |
|             connector.getActivityPoster().postFileFolderAdded(nodeRef);
 | |
| 
 | |
|             return connector.createObjectId(nodeRef);
 | |
|         }
 | |
|         catch (FileNotFoundException e)
 | |
|         {
 | |
|             throw new CmisContentAlreadyExistsException("An object with this name already exists!", e);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public String createPolicy(
 | |
|             String repositoryId, Properties properties, String folderId, List<String> policies,
 | |
|             Acl addAces, Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get the parent folder
 | |
|         getOrCreateFolderInfo(folderId, "Parent Folder");
 | |
| 
 | |
|         String objectTypeId = connector.getObjectTypeIdProperty(properties);
 | |
|         connector.getTypeForCreate(objectTypeId, BaseTypeId.CMIS_POLICY);
 | |
| 
 | |
|         // we should never get here - policies are not creatable!
 | |
|         throw new CmisRuntimeException("Polcies cannot be created!");
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public String createRelationship(
 | |
|             String repositoryId, Properties properties, List<String> policies, Acl addAces,
 | |
|             Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get type
 | |
|         String objectTypeId = connector.getObjectTypeIdProperty(properties);
 | |
|         final TypeDefinitionWrapper type = connector.getTypeForCreate(objectTypeId, BaseTypeId.CMIS_RELATIONSHIP);
 | |
| 
 | |
|         // get source object
 | |
|         String sourceId = connector.getSourceIdProperty(properties);
 | |
|         CMISNodeInfo sourceInfo = getOrCreateNodeInfo(sourceId, "Source");
 | |
| 
 | |
|         if (!sourceInfo.isVariant(CMISObjectVariant.CURRENT_VERSION) && !sourceInfo.isVariant(CMISObjectVariant.FOLDER))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Source is not the latest version of a document or a folder object!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef sourceNodeRef = sourceInfo.getNodeRef();
 | |
| 
 | |
|         // get target object
 | |
|         String targetId = connector.getTargetIdProperty(properties);
 | |
|         CMISNodeInfo targetInfo = getOrCreateNodeInfo(targetId, "Target");
 | |
| 
 | |
|         if (!targetInfo.isVariant(CMISObjectVariant.CURRENT_VERSION) && !targetInfo.isVariant(CMISObjectVariant.FOLDER))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException(
 | |
|                     "Target is not the latest version of a document or a folder object!!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef targetNodeRef = targetInfo.getNodeRef();
 | |
| 
 | |
|         // check policies and ACLs
 | |
|         if ((policies != null) && (!policies.isEmpty()))
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not policy controllable!");
 | |
|         }
 | |
| 
 | |
|         if ((addAces != null) && (addAces.getAces() != null) && (!addAces.getAces().isEmpty()))
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not ACL controllable!");
 | |
|         }
 | |
| 
 | |
|         if ((removeAces != null) && (removeAces.getAces() != null) && (!removeAces.getAces().isEmpty()))
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not ACL controllable!");
 | |
|         }
 | |
| 
 | |
|         // create relationship
 | |
|         // ALF-10085 : disable auditing behaviour for this use case
 | |
|         boolean wasEnabled = connector.disableBehaviour(ContentModel.ASPECT_AUDITABLE, sourceNodeRef);        // Lasts for txn
 | |
|         try
 | |
|         {
 | |
|             AssociationRef assocRef = connector.getNodeService().createAssociation(
 | |
|                     sourceNodeRef, targetNodeRef, type.getAlfrescoClass());
 | |
| 
 | |
|             return CMISConnector.ASSOC_ID_PREFIX + assocRef.getId();
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             if(wasEnabled)
 | |
|             {
 | |
|                 connector.enableBehaviour(ContentModel.ASPECT_AUDITABLE, sourceNodeRef);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private String parseMimeType(ContentStream contentStream)
 | |
|     {
 | |
|     	String mimeType = null;
 | |
| 
 | |
|     	String tmp = contentStream.getMimeType();
 | |
|     	if(tmp != null)
 | |
|     	{
 | |
|     		int idx = tmp.indexOf(";");
 | |
|     		if(idx != -1)
 | |
|     		{
 | |
|     			mimeType = tmp.substring(0, idx).trim();
 | |
|     		}
 | |
|     		else
 | |
|     		{
 | |
|     			mimeType = tmp;
 | |
|     		}
 | |
|     	}
 | |
| 
 | |
|     	return mimeType;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void setContentStream(
 | |
|             String repositoryId, Holder<String> objectId, Boolean overwriteFlag,
 | |
|             Holder<String> changeToken, final ContentStream contentStream, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         if (!info.isVariant(CMISObjectVariant.CURRENT_VERSION) && !info.isVariant(CMISObjectVariant.PWC))
 | |
|         {
 | |
|             throw new CmisStreamNotSupportedException("Content can only be set ondocuments!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         if (((DocumentTypeDefinition) info.getType().getTypeDefinition(false)).getContentStreamAllowed() == ContentStreamAllowed.NOTALLOWED)
 | |
|         {
 | |
|             throw new CmisStreamNotSupportedException("Document type doesn't allow content!");
 | |
|         }
 | |
| 
 | |
|         boolean existed = connector.getContentService().getReader(nodeRef, ContentModel.PROP_CONTENT) != null;
 | |
|         if (existed && !overwriteFlag)
 | |
|         {
 | |
|             throw new CmisContentAlreadyExistsException("Content already exists!");
 | |
|         }
 | |
| 
 | |
|         if ((contentStream == null) || (contentStream.getStream() == null))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("No content!");
 | |
|         }
 | |
| 
 | |
|         // copy stream to temp file
 | |
|         final File tempFile = copyToTempFile(contentStream);
 | |
|         final Charset encoding = getEncoding(tempFile, contentStream.getMimeType());
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             ContentWriter writer = connector.getFileFolderService().getWriter(nodeRef);
 | |
|             String mimeType = parseMimeType(contentStream);
 | |
|             writer.setMimetype(mimeType);
 | |
|             writer.setEncoding(encoding.name());
 | |
|             writer.putContent(tempFile);
 | |
|         }
 | |
|         finally
 | |
|         {
 | |
|             removeTempFile(tempFile);
 | |
|         }
 | |
| 
 | |
|         objectId.setValue(connector.createObjectId(nodeRef));
 | |
| 
 | |
|         connector.getActivityPoster().postFileFolderUpdated(info.isFolder(), nodeRef);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void deleteContentStream(
 | |
|             String repositoryId, Holder<String> objectId, Holder<String> changeToken,
 | |
|             ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         if (!info.isVariant(CMISObjectVariant.CURRENT_VERSION) && !info.isVariant(CMISObjectVariant.PWC))
 | |
|         {
 | |
|             throw new CmisStreamNotSupportedException("Content can only be deleted from ondocuments!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         if (((DocumentTypeDefinition) info.getType().getTypeDefinition(false)).getContentStreamAllowed() == ContentStreamAllowed.REQUIRED)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Document type requires content!");
 | |
|         }
 | |
| 
 | |
|         connector.getNodeService().setProperty(nodeRef, ContentModel.PROP_CONTENT, null);
 | |
| 
 | |
|         connector.getActivityPoster().postFileFolderUpdated(info.isFolder(), nodeRef);
 | |
| 
 | |
|         objectId.setValue(connector.createObjectId(nodeRef));
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void moveObject(
 | |
|             String repositoryId, Holder<String> objectId, String targetFolderId, String sourceFolderId,
 | |
|             ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get object and source and target parent
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         final NodeRef nodeRef = info.getCurrentNodeNodeRef();
 | |
|         final CMISNodeInfo sourceInfo = getOrCreateFolderInfo(sourceFolderId, "Source Folder");
 | |
|         final CMISNodeInfo targetInfo = getOrCreateFolderInfo(targetFolderId, "Target Folder");
 | |
| 
 | |
|         connector.checkChildObjectType(targetInfo, info.getType().getTypeId());
 | |
| 
 | |
|         ChildAssociationRef primaryParentRef = connector.getNodeService().getPrimaryParent(nodeRef);
 | |
|         // if this is a primary child node, move it
 | |
|         if (primaryParentRef.getParentRef().equals(sourceInfo.getNodeRef()))
 | |
|         {
 | |
|             connector.getNodeService().moveNode(
 | |
|                     nodeRef, targetInfo.getNodeRef(),
 | |
|                     primaryParentRef.getTypeQName(), primaryParentRef.getQName());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             boolean found = false;
 | |
|             // otherwise, reparent it
 | |
|             for (ChildAssociationRef parent : connector.getNodeService().getParentAssocs(
 | |
|                     nodeRef, ContentModel.ASSOC_CONTAINS, RegexQNamePattern.MATCH_ALL))
 | |
|             {
 | |
|                 if (parent.getParentRef().equals(sourceInfo.getNodeRef()))
 | |
|                 {
 | |
|                     connector.getNodeService().removeChildAssociation(parent);
 | |
|                     connector.getNodeService().addChild(
 | |
|                             targetInfo.getNodeRef(), nodeRef,
 | |
|                             ContentModel.ASSOC_CONTAINS, parent.getQName());
 | |
|                     found = true;
 | |
|                 }
 | |
|             }
 | |
|             if (!found)
 | |
|             {
 | |
|                 throw new IllegalArgumentException(
 | |
|                         new CMISInvalidArgumentException(
 | |
|                                 "Document is not a child of the source folder that was specified!"));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void updateProperties(
 | |
|             String repositoryId, Holder<String> objectId, Holder<String> changeToken,
 | |
|             final Properties properties, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         final CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Relationship properties cannot be updated!");
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (info.isVariant(CMISObjectVariant.VERSION))
 | |
|             {
 | |
|                 throw new CmisInvalidArgumentException("Document is not the latest version!");
 | |
|             }
 | |
| 
 | |
|             final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|             connector.setProperties(nodeRef, info.getType(), properties, new String[0]);
 | |
| 
 | |
|             objectId.setValue(connector.createObjectId(nodeRef));
 | |
| 
 | |
|             if (getContext().isObjectInfoRequired())
 | |
|             {
 | |
|                 getObjectInfo(repositoryId, objectId.getValue(), "*", IncludeRelationships.NONE);
 | |
|             }
 | |
| 
 | |
|             connector.getActivityPoster().postFileFolderUpdated(info.isFolder(), nodeRef);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void deleteObject(String repositoryId, String objectId, Boolean allVersions, ExtensionsData extension)
 | |
|     {
 | |
|         deleteObjectOrCancelCheckOut(repositoryId, objectId, allVersions, extension);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void deleteObjectOrCancelCheckOut(
 | |
|             String repositoryId, final String objectId, final Boolean allVersions,
 | |
|             ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         final CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         do
 | |
|         {
 | |
|             // handle relationships
 | |
|             if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|             {
 | |
|                 AssociationRef assocRef = info.getAssociationRef();
 | |
|                 connector.getNodeService().removeAssociation(
 | |
|                         assocRef.getSourceRef(),
 | |
|                         assocRef.getTargetRef(),
 | |
|                         assocRef.getTypeQName());
 | |
|                 break;      // Reason for do-while
 | |
|             }
 | |
| 
 | |
|             NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|             // handle PWC
 | |
|             if (info.isVariant(CMISObjectVariant.PWC))
 | |
|             {
 | |
|                 connector.getCheckOutCheckInService().cancelCheckout(nodeRef);
 | |
|                 break;      // Reason for do-while
 | |
|             }
 | |
| 
 | |
|             // handle folders
 | |
|             if (info.isFolder())
 | |
|             {
 | |
|                 if (connector.getNodeService().getChildAssocs(nodeRef, ContentModel.ASSOC_CONTAINS, RegexQNamePattern.MATCH_ALL).size() > 0)
 | |
|                 {
 | |
|                     throw new CmisConstraintException(
 | |
|                             "Could not delete folder with at least one child!");
 | |
|                 }
 | |
| 
 | |
|                 connector.deleteNode(nodeRef, true);
 | |
|                 break;      // Reason for do-while
 | |
|             }
 | |
| 
 | |
|             if(info.hasPWC())
 | |
|             {
 | |
|                 // is a checked out document. If a delete, don't allow unless checkout is canceled. If a cancel
 | |
|                 // checkout, not allowed.
 | |
|                 throw new CmisConstraintException(
 | |
|                 "Could not delete/cancel checkout on the original checked out document");
 | |
|             }
 | |
| 
 | |
|             // handle versions
 | |
|             if (allVersions)
 | |
|             {
 | |
|                 NodeRef workingCopy = connector.getCheckOutCheckInService().getWorkingCopy(nodeRef);
 | |
|                 if (workingCopy != null)
 | |
|                 {
 | |
|                     connector.getCheckOutCheckInService().cancelCheckout(workingCopy);
 | |
|                 }
 | |
|             }
 | |
|             else if (info.isVariant(CMISObjectVariant.VERSION))
 | |
|             {
 | |
|                 Version version = ((CMISNodeInfoImpl) info).getVersion();
 | |
|                 connector.getVersionService().deleteVersion(nodeRef, version);
 | |
|                 break;      // Reason for do-while
 | |
|             }
 | |
| 
 | |
|             if (info.isVariant(CMISObjectVariant.VERSION))
 | |
|             {
 | |
|                 nodeRef = info.getCurrentNodeNodeRef();
 | |
|             }
 | |
| 
 | |
|             // attempt to delete the node
 | |
|             if (allVersions)
 | |
|             {
 | |
|                 connector.deleteNode(nodeRef, true);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 CMISNodeInfoImpl infoImpl = ((CMISNodeInfoImpl) info);
 | |
|                 Version version = infoImpl.getVersion();
 | |
| 
 | |
|                 if (infoImpl.getVersionHistory().getPredecessor(version) == null)
 | |
|                 {
 | |
|                     connector.deleteNode(nodeRef, true);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     connector.getVersionService().deleteVersion(nodeRef, version);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         while (false);      // Dodgey, but avoided having to play with too much code during refactor
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public FailedToDeleteData deleteTree(
 | |
|             String repositoryId, String folderId, Boolean allVersions,
 | |
|             UnfileObject unfileObjects, final Boolean continueOnFailure, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (!allVersions)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Only allVersions=true supported!");
 | |
|         }
 | |
| 
 | |
|         if (unfileObjects == UnfileObject.UNFILE)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Unfiling not supported!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef folderNodeRef = getOrCreateFolderInfo(folderId, "Folder").getNodeRef();
 | |
|         final FailedToDeleteDataImpl result = new FailedToDeleteDataImpl();
 | |
| 
 | |
|         result.setIds(deleteBranch(folderNodeRef, continueOnFailure));
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     private List<String> deleteBranch(NodeRef nodeRef, boolean continueOnFailure)
 | |
|     {
 | |
|         List<String> result = new ArrayList<String>();
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             // remove children
 | |
|             List<ChildAssociationRef> childrenList = connector.getNodeService().getChildAssocs(nodeRef);
 | |
|             if (childrenList != null)
 | |
|             {
 | |
|                 for (ChildAssociationRef child : childrenList)
 | |
|                 {
 | |
|                     List<String> ftod = deleteBranch(child.getChildRef(), continueOnFailure);
 | |
|                     if (!ftod.isEmpty())
 | |
|                     {
 | |
|                         result.addAll(ftod);
 | |
|                         if (!continueOnFailure)
 | |
|                         {
 | |
|                             return result;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // remove not primary parent associations
 | |
|             // TODO: This can be removed once we fix node archival
 | |
|             List<ChildAssociationRef> childAssociations = connector.getNodeService().getParentAssocs(nodeRef);
 | |
|             if (childAssociations != null)
 | |
|             {
 | |
|                 for (ChildAssociationRef childAssoc : childAssociations)
 | |
|                 {
 | |
|                     if (!childAssoc.isPrimary())
 | |
|                     {
 | |
|                         connector.getNodeService().removeChildAssociation(childAssoc);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // attempt to delete the node
 | |
|             connector.deleteNode(nodeRef, true);
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             result.add(nodeRef.getId());
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectData getObject(
 | |
|             String repositoryId, String objectId, String filter, Boolean includeAllowableActions,
 | |
|             IncludeRelationships includeRelationships, String renditionFilter, Boolean includePolicyIds,
 | |
|             Boolean includeAcl, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // create a CMIS object
 | |
|         ObjectData object = connector.createCMISObject(
 | |
|                 info, filter, includeAllowableActions, includeRelationships,
 | |
|                 renditionFilter, includePolicyIds, includeAcl);
 | |
| 
 | |
|         if (getContext().isObjectInfoRequired())
 | |
|         {
 | |
|             getObjectInfo(repositoryId, info.getObjectId(), includeRelationships);
 | |
|         }
 | |
| 
 | |
|         return object;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectData getObjectByPath(
 | |
|             String repositoryId, String path, String filter, Boolean includeAllowableActions,
 | |
|             IncludeRelationships includeRelationships, String renditionFilter, Boolean includePolicyIds,
 | |
|             Boolean includeAcl, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // start at the root node
 | |
|         NodeRef rootNodeRef = connector.getRootNodeRef();
 | |
| 
 | |
|         if (path.equals("/"))
 | |
|         {
 | |
|             return connector.createCMISObject(createNodeInfo(rootNodeRef), filter, includeAllowableActions,
 | |
|                     includeRelationships, renditionFilter, includePolicyIds, includeAcl);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 // resolve path and get the node ref
 | |
|                 FileInfo fileInfo = connector.getFileFolderService().resolveNamePath(
 | |
|                         rootNodeRef,
 | |
|                         Arrays.asList(path.substring(1).split("/")));
 | |
| 
 | |
|                 if(connector.filter(fileInfo.getNodeRef()))
 | |
|                 {
 | |
|                     throw new CmisObjectNotFoundException("Object not found: " + path);
 | |
|                 }
 | |
| 
 | |
|                 CMISNodeInfo info = createNodeInfo(fileInfo.getNodeRef());
 | |
| 
 | |
|                 ObjectData object = connector.createCMISObject(
 | |
|                         info, fileInfo, filter, includeAllowableActions,
 | |
|                         includeRelationships, renditionFilter, includePolicyIds, includeAcl);
 | |
| 
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, info.getObjectId(), includeRelationships);
 | |
|                 }
 | |
| 
 | |
|                 return object;
 | |
|             }
 | |
|             catch (FileNotFoundException e)
 | |
|             {
 | |
|                 throw new CmisObjectNotFoundException("Object not found: " + path);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Properties getProperties(String repositoryId, String objectId, String filter, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         if (getContext().isObjectInfoRequired())
 | |
|         {
 | |
|             getObjectInfo(repositoryId, info.getObjectId(), IncludeRelationships.NONE);
 | |
|         }
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             return connector.getAssocProperties(info, filter);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return connector.getNodeProperties(info, filter);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public AllowableActions getAllowableActions(String repositoryId, String objectId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         return connector.getAllowableActions(info);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ContentStream getContentStream(
 | |
|             String repositoryId, String objectId, String streamId, BigInteger offset,
 | |
|             BigInteger length, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // relationships cannot have content
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object is a relationship and cannot have content!");
 | |
|         }
 | |
| 
 | |
|         // now get it
 | |
|         return connector.getContentStream(info, streamId, offset, length);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<RenditionData> getRenditions(String repositoryId, String objectId, String renditionFilter,
 | |
|             BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             return Collections.emptyList();
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return connector.getRenditions(info.getNodeRef(), renditionFilter, maxItems, skipCount);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // --- versioning service ---
 | |
| 
 | |
|     @Override
 | |
|     public void checkOut(
 | |
|             String repositoryId, final Holder<String> objectId, ExtensionsData extension,
 | |
|             final Holder<Boolean> contentCopied)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         // Check for current version
 | |
|         if (info.isVariant(CMISObjectVariant.CURRENT_VERSION))
 | |
|         {
 | |
|             // Good
 | |
|         }
 | |
|         else if (info.isVariant(CMISObjectVariant.VERSION))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Can't check out an old version of a document");
 | |
|         }
 | |
|         else {   
 | |
|             throw new CmisInvalidArgumentException("Only documents can be checked out! Object was a " + info.getObjectVariant().toString());
 | |
|         }
 | |
| 
 | |
|         // get object
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         if (!((DocumentTypeDefinition) info.getType().getTypeDefinition(false)).isVersionable())
 | |
|         {
 | |
|             throw new CmisConstraintException("Document is not versionable!");
 | |
|         }
 | |
| 
 | |
|         // check out
 | |
|         NodeRef pwcNodeRef = connector.getCheckOutCheckInService().checkout(nodeRef);
 | |
|         CMISNodeInfo pwcNodeInfo = createNodeInfo(pwcNodeRef);
 | |
|         objectId.setValue(pwcNodeInfo.getObjectId());
 | |
| 
 | |
|         if (contentCopied != null)
 | |
|         {
 | |
|             contentCopied.setValue(connector.getFileFolderService().getReader(pwcNodeRef) != null);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void cancelCheckOut(String repositoryId, String objectId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // only accept a PWC
 | |
|         if (!info.isVariant(CMISObjectVariant.PWC))
 | |
|         {
 | |
|             throw new CmisVersioningException("Object is not a PWC!");
 | |
|         }
 | |
| 
 | |
|         // get object
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         // cancel check out
 | |
|         connector.getCheckOutCheckInService().cancelCheckout(nodeRef);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void checkIn(
 | |
|             String repositoryId, final Holder<String> objectId, final Boolean major,
 | |
|             final Properties properties, final ContentStream contentStream, final String checkinComment,
 | |
|             final List<String> policies, final Acl addAces, final Acl removeAces, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId.getValue(), "Object");
 | |
| 
 | |
|         // only accept a PWC
 | |
|         if (!info.isVariant(CMISObjectVariant.PWC))
 | |
|         {
 | |
|             throw new CmisVersioningException("Object is not a PWC!");
 | |
|         }
 | |
| 
 | |
|         // get object
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
|         final TypeDefinitionWrapper type = info.getType();
 | |
| 
 | |
|         // copy stream to temp file
 | |
|         final File tempFile = copyToTempFile(contentStream);
 | |
|         final Charset encoding = (tempFile == null ? null : getEncoding(tempFile, contentStream.getMimeType()));
 | |
| 
 | |
|         // check in
 | |
|         // update PWC
 | |
|         connector.setProperties(nodeRef, type, properties,
 | |
|                 new String[] { PropertyIds.OBJECT_TYPE_ID });
 | |
|         connector.applyPolicies(nodeRef, type, policies);
 | |
|         connector.applyACL(nodeRef, type, addAces, removeAces);
 | |
| 
 | |
|         // handle content
 | |
|         if (contentStream != null)
 | |
|         {
 | |
|             // write content
 | |
|             ContentWriter writer = connector.getFileFolderService().getWriter(nodeRef);
 | |
|             writer.setMimetype(parseMimeType(contentStream));
 | |
|             writer.setEncoding(encoding.name());
 | |
|             writer.putContent(tempFile);
 | |
|         }
 | |
| 
 | |
|         // check aspect
 | |
|         if (connector.getNodeService().hasAspect(nodeRef, ContentModel.ASPECT_VERSIONABLE) == false)
 | |
|         {
 | |
|             Map<QName, Serializable> props = new HashMap<QName, Serializable>();
 | |
|             props.put(ContentModel.PROP_INITIAL_VERSION, false);
 | |
|             props.put(ContentModel.PROP_AUTO_VERSION, false);
 | |
|             connector.getNodeService().addAspect(nodeRef, ContentModel.ASPECT_VERSIONABLE, props);
 | |
|         }
 | |
| 
 | |
|         // create version properties
 | |
|         Map<String, Serializable> versionProperties = new HashMap<String, Serializable>(5);
 | |
|         versionProperties.put(VersionModel.PROP_VERSION_TYPE, major ? VersionType.MAJOR
 | |
|                 : VersionType.MINOR);
 | |
|         if (checkinComment != null)
 | |
|         {
 | |
|             versionProperties.put(VersionModel.PROP_DESCRIPTION, checkinComment);
 | |
|         }
 | |
| 
 | |
|         // check in
 | |
|         NodeRef newNodeRef = connector.getCheckOutCheckInService().checkin(nodeRef, versionProperties);
 | |
| 
 | |
|         connector.getActivityPoster().postFileFolderUpdated(info.isFolder(), newNodeRef);
 | |
| 
 | |
|         objectId.setValue(connector.createObjectId(newNodeRef));
 | |
| 
 | |
|         removeTempFile(tempFile);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<ObjectData> getAllVersions(
 | |
|             String repositoryId, String objectId, String versionSeriesId, String filter,
 | |
|             Boolean includeAllowableActions, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (versionSeriesId == null && objectId != null)
 | |
|         {
 | |
|             // it's a browser binding call
 | |
|             versionSeriesId = connector.getCurrentVersionId(objectId);
 | |
|         }
 | |
| 
 | |
|         if (versionSeriesId == null)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object Id or Object Series Id must be set!");
 | |
|         }
 | |
| 
 | |
|         List<ObjectData> result = new ArrayList<ObjectData>();
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(versionSeriesId, "Version Series");
 | |
| 
 | |
|         if (!info.isVariant(CMISObjectVariant.CURRENT_VERSION))
 | |
|         {
 | |
|             // the version series id is the id of current version, which is a
 | |
|             // document
 | |
|             throw new CmisInvalidArgumentException("Version Series does not exist!");
 | |
|         }
 | |
| 
 | |
|         // get current version and it's history
 | |
|         NodeRef nodeRef = info.getNodeRef();
 | |
|         VersionHistory versionHistory = ((CMISNodeInfoImpl) info).getVersionHistory();
 | |
| 
 | |
| //      Collection<Version> versions = versionHistory.getAllVersions();
 | |
| //      if (versionHistory == null || versions.size() == 0  || versions.size() == 1 && versions.contains(versionHistory.getHeadVersion()))
 | |
|         if (versionHistory == null)
 | |
|         {
 | |
|             // add current version
 | |
|             result.add(connector.createCMISObject(info, filter, includeAllowableActions, IncludeRelationships.NONE,
 | |
|                     CMISConnector.RENDITION_NONE, false, false));
 | |
| 
 | |
|             if (getContext().isObjectInfoRequired())
 | |
|             {
 | |
|                 getObjectInfo(repositoryId, info.getObjectId(), IncludeRelationships.NONE);
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (info.hasPWC())
 | |
|             {
 | |
|                 CMISNodeInfo pwcInfo = createNodeInfo(connector.getCheckOutCheckInService().getWorkingCopy(nodeRef));
 | |
| 
 | |
|                 result.add(
 | |
|                         connector.createCMISObject(
 | |
|                                 pwcInfo, filter, includeAllowableActions,
 | |
|                                 IncludeRelationships.NONE, CMISConnector.RENDITION_NONE, false, false));
 | |
| 
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, pwcInfo.getObjectId(), IncludeRelationships.NONE);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // convert the version history
 | |
|             for (Version version : versionHistory.getAllVersions())
 | |
|             {
 | |
|                 CMISNodeInfo versionInfo = createNodeInfo(version.getFrozenStateNodeRef(), versionHistory);
 | |
| 
 | |
|                 result.add(
 | |
|                         connector.createCMISObject(
 | |
|                                 versionInfo, filter, includeAllowableActions,
 | |
|                                 IncludeRelationships.NONE, CMISConnector.RENDITION_NONE, false, false));
 | |
| 
 | |
|                 if (getContext().isObjectInfoRequired())
 | |
|                 {
 | |
|                     getObjectInfo(repositoryId, versionInfo.getObjectId(), IncludeRelationships.NONE);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectData getObjectOfLatestVersion(
 | |
|             String repositoryId, String objectId, String versionSeriesId,
 | |
|             Boolean major, String filter, Boolean includeAllowableActions, IncludeRelationships includeRelationships,
 | |
|             String renditionFilter, Boolean includePolicyIds, Boolean includeAcl, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (objectId != null)
 | |
|         {
 | |
|             // it's an AtomPub call
 | |
|             versionSeriesId = connector.getCurrentVersionId(objectId);
 | |
|         }
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(versionSeriesId, "Version Series");
 | |
|         CMISNodeInfo versionInfo = createNodeInfo(((CMISNodeInfoImpl) info).getLatestVersionNodeRef(major));
 | |
| 
 | |
|         ObjectData object = connector.createCMISObject(
 | |
|                 versionInfo, filter, includeAllowableActions,
 | |
|                 includeRelationships, renditionFilter, includePolicyIds, includeAcl);
 | |
| 
 | |
|         if (getContext().isObjectInfoRequired())
 | |
|         {
 | |
|             getObjectInfo(repositoryId, info.getObjectId(), includeRelationships);
 | |
|         }
 | |
| 
 | |
|         return object;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Properties getPropertiesOfLatestVersion(
 | |
|             String repositoryId, String objectId, String versionSeriesId,
 | |
|             Boolean major, String filter, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (objectId != null)
 | |
|         {
 | |
|             // it's an AtomPub call
 | |
|             versionSeriesId = connector.getCurrentVersionId(objectId);
 | |
|         }
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(versionSeriesId, "Version Series");
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             return connector.getAssocProperties(info, filter);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             CMISNodeInfo versionInfo = createNodeInfo(((CMISNodeInfoImpl) info).getLatestVersionNodeRef(major));
 | |
|             addNodeInfo(versionInfo);
 | |
|             return connector.getNodeProperties(versionInfo, filter);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // --- multifiling service ---
 | |
| 
 | |
|     @Override
 | |
|     public void addObjectToFolder(
 | |
|             String repositoryId, String objectId, String folderId, Boolean allVersions,
 | |
|             ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (!allVersions)
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Only allVersions=true supported!");
 | |
|         }
 | |
| 
 | |
|         // get node ref
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         if (!info.isDocument())
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object is not a document!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         // get the folder node ref
 | |
|         final CMISNodeInfo folderInfo = getOrCreateFolderInfo(folderId, "Folder");
 | |
| 
 | |
|         connector.checkChildObjectType(folderInfo, info.getType().getTypeId());
 | |
| 
 | |
|         final QName name = QName.createQName(
 | |
|                 NamespaceService.CONTENT_MODEL_1_0_URI,
 | |
|                 QName.createValidLocalName((String) connector.getNodeService().getProperty(nodeRef,
 | |
|                         ContentModel.PROP_NAME)));
 | |
| 
 | |
|         connector.getNodeService().addChild(
 | |
|                 folderInfo.getNodeRef(), nodeRef, ContentModel.ASSOC_CONTAINS, name);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void removeObjectFromFolder(String repositoryId, String objectId, String folderId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // get node ref
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         if (!info.isDocument())
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object is not a document!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getNodeRef();
 | |
| 
 | |
|         // get the folder node ref
 | |
|         final NodeRef folderNodeRef = getOrCreateFolderInfo(folderId, "Folder").getNodeRef();
 | |
| 
 | |
|         // check primary parent
 | |
|         if (connector.getNodeService().getPrimaryParent(nodeRef).getParentRef().equals(folderNodeRef))
 | |
|         {
 | |
|             throw new CmisConstraintException(
 | |
|                     "Unfiling from primary parent folder is not supported! Use deleteObject() instead.");
 | |
|         }
 | |
| 
 | |
|         connector.getNodeService().removeChild(folderNodeRef, nodeRef);
 | |
|     }
 | |
| 
 | |
|     // --- discovery service ---
 | |
| 
 | |
|     @Override
 | |
|     public ObjectList getContentChanges(
 | |
|             String repositoryId, Holder<String> changeLogToken, Boolean includeProperties,
 | |
|             String filter, Boolean includePolicyIds, Boolean includeAcl, BigInteger maxItems, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         return connector.getContentChanges(changeLogToken, maxItems);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public ObjectList query(String repositoryId, String statement, Boolean searchAllVersions,
 | |
|             Boolean includeAllowableActions, IncludeRelationships includeRelationships, String renditionFilter,
 | |
|             BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         if (searchAllVersions.booleanValue())
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Search all version is not supported!");
 | |
|         }
 | |
| 
 | |
|         return connector.query(
 | |
|                 statement, includeAllowableActions, includeRelationships, renditionFilter,
 | |
|                 maxItems, skipCount);
 | |
|     }
 | |
| 
 | |
|     // --- relationship service ---
 | |
| 
 | |
|     @Override
 | |
|     public ObjectList getObjectRelationships(
 | |
|             String repositoryId, String objectId, Boolean includeSubRelationshipTypes,
 | |
|             RelationshipDirection relationshipDirection, String typeId, String filter, Boolean includeAllowableActions,
 | |
|             BigInteger maxItems, BigInteger skipCount, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object is a relationship!");
 | |
|         }
 | |
| 
 | |
|         if (info.isVariant(CMISObjectVariant.VERSION))
 | |
|         {
 | |
|             throw new CmisInvalidArgumentException("Object is a document version!");
 | |
|         }
 | |
| 
 | |
|         // check if the relationship base type is requested
 | |
|         if (BaseTypeId.CMIS_RELATIONSHIP.value().equals(typeId))
 | |
|         {
 | |
|             boolean isrt = (includeSubRelationshipTypes == null ? false : includeSubRelationshipTypes.booleanValue());
 | |
|             if (isrt)
 | |
|             {
 | |
|                 // all relationships are a direct subtype of the base type in
 | |
|                 // Alfresco -> remove filter
 | |
|                 typeId = null;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // there are no relationships of the base type in Alfresco ->
 | |
|                 // return empty list
 | |
|                 ObjectListImpl result = new ObjectListImpl();
 | |
|                 result.setHasMoreItems(false);
 | |
|                 result.setNumItems(BigInteger.ZERO);
 | |
|                 result.setObjects(new ArrayList<ObjectData>());
 | |
|                 return result;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return connector.getObjectRelationships(
 | |
|                 info.getNodeRef(), relationshipDirection, typeId, filter, includeAllowableActions,
 | |
|                 maxItems, skipCount);
 | |
|     }
 | |
| 
 | |
|     // --- policy service ---
 | |
| 
 | |
|     @Override
 | |
|     public void applyPolicy(String repositoryId, String policyId, String objectId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         TypeDefinitionWrapper type = info.getType();
 | |
|         if (type == null)
 | |
|         {
 | |
|             throw new CmisObjectNotFoundException("No corresponding type found! Not a CMIS object?");
 | |
|         }
 | |
| 
 | |
|         connector.applyPolicies(info.getNodeRef(), type, Collections.singletonList(policyId));
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void removePolicy(String repositoryId, String policyId, String objectId, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         TypeDefinitionWrapper type = info.getType();
 | |
|         if (type == null)
 | |
|         {
 | |
|             throw new CmisObjectNotFoundException("No corresponding type found! Not a CMIS object?");
 | |
|         }
 | |
| 
 | |
|         throw new CmisConstraintException("Object is not policy controllable!");
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public List<ObjectData> getAppliedPolicies(
 | |
|             String repositoryId, String objectId, String filter, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // what kind of object is it?
 | |
|         getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // policies are not supported -> return empty list
 | |
|         return Collections.emptyList();
 | |
|     }
 | |
| 
 | |
|     // --- ACL service ---
 | |
| 
 | |
|     @Override
 | |
|     public Acl applyAcl(
 | |
|             String repositoryId, String objectId, final Acl addAces, final Acl removeAces,
 | |
|             AclPropagation aclPropagation, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // We are spec compliant if we just let it through and the tck will not fail
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // relationships don't have ACLs
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not ACL controllable!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getCurrentNodeNodeRef();
 | |
|         final TypeDefinitionWrapper type = info.getType();
 | |
| 
 | |
|         connector.applyACL(nodeRef, type, addAces, removeAces);
 | |
| 
 | |
|         return connector.getACL(nodeRef, false);
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public Acl applyAcl(String repositoryId, String objectId, final Acl aces, AclPropagation aclPropagation)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         // We are spec compliant if we just let it through and the tck will not fail
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // relationships don't have ACLs
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             throw new CmisConstraintException("Relationships are not ACL controllable!");
 | |
|         }
 | |
| 
 | |
|         final NodeRef nodeRef = info.getCurrentNodeNodeRef();
 | |
|         final TypeDefinitionWrapper type = info.getType();
 | |
| 
 | |
|         connector.applyACL(nodeRef, type, aces);
 | |
| 
 | |
|         return connector.getACL(nodeRef, false);
 | |
|     }
 | |
| 
 | |
|     @SuppressWarnings("unchecked")
 | |
|     @Override
 | |
|     public Acl getAcl(String repositoryId, String objectId, Boolean onlyBasicPermissions, ExtensionsData extension)
 | |
|     {
 | |
|         checkRepositoryId(repositoryId);
 | |
| 
 | |
|         CMISNodeInfo info = getOrCreateNodeInfo(objectId, "Object");
 | |
| 
 | |
|         // relationships don't have ACLs
 | |
|         if (info.isVariant(CMISObjectVariant.ASSOC))
 | |
|         {
 | |
|             return new AccessControlListImpl(Collections.EMPTY_LIST);
 | |
|         }
 | |
| 
 | |
|         // get the ACL
 | |
|         return connector.getACL(info.getCurrentNodeNodeRef(), onlyBasicPermissions);
 | |
|     }
 | |
| 
 | |
|     // --------------------------------------------------------
 | |
| 
 | |
|     /**
 | |
|      * Collects the {@link ObjectInfo} about an object.
 | |
|      * 
 | |
|      * (Provided by OpenCMIS, but optimized for Alfresco.)
 | |
|      */
 | |
|     @Override
 | |
|     public ObjectInfo getObjectInfo(String repositoryId, String objectId)
 | |
|     {
 | |
|         return getObjectInfo(repositoryId, objectId, null, IncludeRelationships.BOTH);
 | |
|     }
 | |
| 
 | |
|     protected ObjectInfo getObjectInfo(String repositoryId, String objectId, IncludeRelationships includeRelationships)
 | |
|     {
 | |
|         return getObjectInfo(repositoryId, objectId, null, includeRelationships);
 | |
|     }
 | |
| 
 | |
|     protected ObjectInfo getObjectInfo(
 | |
|             String repositoryId, String objectId, String filter,
 | |
|             IncludeRelationships includeRelationships)
 | |
|     {
 | |
|         ObjectInfo info = objectInfoMap.get(objectId);
 | |
|         if (info == null)
 | |
|         {
 | |
|             CMISNodeInfo nodeInfo = getOrCreateNodeInfo(objectId);
 | |
| 
 | |
|             if (nodeInfo.getObjectVariant() == CMISObjectVariant.INVALID_ID
 | |
|                     || nodeInfo.getObjectVariant() == CMISObjectVariant.NOT_EXISTING
 | |
|                     || nodeInfo.getObjectVariant() == CMISObjectVariant.NOT_A_CMIS_OBJECT
 | |
|                     || nodeInfo.getObjectVariant() == CMISObjectVariant.PERMISSION_DENIED)
 | |
|             {
 | |
|                 info = null;
 | |
|             } else
 | |
|             {
 | |
|                 // object info has not been found -> create one
 | |
|                 try
 | |
|                 {
 | |
|                     if (filter == null)
 | |
|                     {
 | |
|                         filter = MIN_FILTER;
 | |
|                     }
 | |
|                     else if (!filter.equals("*"))
 | |
|                     {
 | |
|                         filter = filter + "," + MIN_FILTER;
 | |
|                     }
 | |
| 
 | |
|                     // get the object and its info
 | |
|                     ObjectData object = connector.createCMISObject(
 | |
|                             nodeInfo, filter, false, includeRelationships, null, false, false);
 | |
| 
 | |
|                     info = getObjectInfoIntern(repositoryId, object);
 | |
| 
 | |
|                     // add object info
 | |
|                     objectInfoMap.put(objectId, info);
 | |
|                 }
 | |
|                 catch (Exception e)
 | |
|                 {
 | |
|                     e.printStackTrace();
 | |
|                     info = null;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return info;
 | |
|     }
 | |
| 
 | |
|     protected String getGuid(String nodeId)
 | |
|     {
 | |
|         int idx = nodeId.lastIndexOf("/");
 | |
|         if(idx != -1)
 | |
|         {
 | |
|             return nodeId.substring(idx+1);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return nodeId;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Collects the {@link ObjectInfo} about an object.
 | |
|      * 
 | |
|      * (Provided by OpenCMIS, but optimized for Alfresco.)
 | |
|      */
 | |
|     @Override
 | |
|     protected ObjectInfo getObjectInfoIntern(String repositoryId, ObjectData object)
 | |
|     {
 | |
|         // if the object has no properties, stop here
 | |
|         if (object.getProperties() == null || object.getProperties().getProperties() == null)
 | |
|         {
 | |
|             throw new CmisRuntimeException("No properties!");
 | |
|         }
 | |
| 
 | |
|         CMISNodeInfo ni = getOrCreateNodeInfo(object.getId());
 | |
| 
 | |
|         ObjectInfoImpl info = new ObjectInfoImpl();
 | |
| 
 | |
|         // general properties
 | |
|         info.setObject(object);
 | |
|         info.setId(object.getId());
 | |
|         info.setName(ni.getName());
 | |
|         info.setCreatedBy(getStringProperty(object, PropertyIds.CREATED_BY));
 | |
|         info.setCreationDate(getDateTimeProperty(object, PropertyIds.CREATION_DATE));
 | |
|         info.setLastModificationDate(getDateTimeProperty(object, PropertyIds.LAST_MODIFICATION_DATE));
 | |
|         info.setTypeId(getIdProperty(object, PropertyIds.OBJECT_TYPE_ID));
 | |
|         info.setBaseType(object.getBaseTypeId());
 | |
| 
 | |
|         if (ni.isRelationship())
 | |
|         {
 | |
|             // versioning
 | |
|             info.setWorkingCopyId(null);
 | |
|             info.setWorkingCopyOriginalId(null);
 | |
| 
 | |
|             info.setVersionSeriesId(null);
 | |
|             info.setIsCurrentVersion(true);
 | |
|             info.setWorkingCopyId(null);
 | |
|             info.setWorkingCopyOriginalId(null);
 | |
| 
 | |
|             // content
 | |
|             info.setHasContent(false);
 | |
|             info.setContentType(null);
 | |
|             info.setFileName(null);
 | |
| 
 | |
|             // parent
 | |
|             info.setHasParent(false);
 | |
| 
 | |
|             // policies and relationships
 | |
|             info.setSupportsRelationships(false);
 | |
|             info.setSupportsPolicies(false);
 | |
| 
 | |
|             // renditions
 | |
|             info.setRenditionInfos(null);
 | |
| 
 | |
|             // relationships
 | |
|             info.setRelationshipSourceIds(null);
 | |
|             info.setRelationshipTargetIds(null);
 | |
| 
 | |
|             // global settings
 | |
|             info.setHasAcl(false);
 | |
|             info.setSupportsDescendants(false);
 | |
|             info.setSupportsFolderTree(false);
 | |
|         }
 | |
|         else if (ni.isFolder())
 | |
|         {
 | |
|             // versioning
 | |
|             info.setWorkingCopyId(null);
 | |
|             info.setWorkingCopyOriginalId(null);
 | |
| 
 | |
|             info.setVersionSeriesId(null);
 | |
|             info.setIsCurrentVersion(true);
 | |
|             info.setWorkingCopyId(null);
 | |
|             info.setWorkingCopyOriginalId(null);
 | |
| 
 | |
|             // content
 | |
|             info.setHasContent(false);
 | |
|             info.setContentType(null);
 | |
|             info.setFileName(null);
 | |
| 
 | |
|             // parent
 | |
|             info.setHasParent(!ni.isRootFolder());
 | |
| 
 | |
|             // policies and relationships
 | |
|             info.setSupportsRelationships(true);
 | |
|             info.setSupportsPolicies(true);
 | |
| 
 | |
|             // renditions
 | |
|             info.setRenditionInfos(null);
 | |
| 
 | |
|             // relationships
 | |
|             setRelaionshipsToObjectInfo(object, info);
 | |
| 
 | |
|             // global settings
 | |
|             info.setHasAcl(true);
 | |
|             info.setSupportsDescendants(true);
 | |
|             info.setSupportsFolderTree(true);
 | |
|         }
 | |
|         else if (ni.isDocument())
 | |
|         {
 | |
|             // versioning
 | |
|             info.setWorkingCopyId(null);
 | |
|             info.setWorkingCopyOriginalId(null);
 | |
| 
 | |
|             info.setVersionSeriesId(getGuid(ni.getCurrentNodeId()));
 | |
| 
 | |
|             if (ni.isPWC())
 | |
|             {
 | |
|                 info.setIsCurrentVersion(false);
 | |
|                 info.setWorkingCopyId(ni.getObjectId());
 | |
|                 info.setWorkingCopyOriginalId(ni.getCurrentObjectId());
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 info.setIsCurrentVersion(ni.isCurrentVersion());
 | |
| 
 | |
|                 if (ni.hasPWC())
 | |
|                 {
 | |
|                     info.setWorkingCopyId(getGuid(ni.getCurrentNodeId()) + CMISConnector.ID_SEPERATOR
 | |
|                             + CMISConnector.PWC_VERSION_LABEL);
 | |
|                     info.setWorkingCopyOriginalId(ni.getCurrentObjectId());
 | |
|                 } else
 | |
|                 {
 | |
|                     info.setWorkingCopyId(null);
 | |
|                     info.setWorkingCopyOriginalId(null);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // content
 | |
|             String fileName = getStringProperty(object, PropertyIds.CONTENT_STREAM_FILE_NAME);
 | |
|             String mimeType = getStringProperty(object, PropertyIds.CONTENT_STREAM_MIME_TYPE);
 | |
|             String streamId = getIdProperty(object, PropertyIds.CONTENT_STREAM_ID);
 | |
|             BigInteger length = getIntegerProperty(object, PropertyIds.CONTENT_STREAM_LENGTH);
 | |
|             boolean hasContent = fileName != null || mimeType != null || streamId != null || length != null;
 | |
|             if (hasContent)
 | |
|             {
 | |
|                 info.setHasContent(hasContent);
 | |
|                 info.setContentType(mimeType);
 | |
|                 info.setFileName(fileName);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 info.setHasContent(false);
 | |
|                 info.setContentType(null);
 | |
|                 info.setFileName(null);
 | |
|             }
 | |
| 
 | |
|             // parent
 | |
|             info.setHasParent(ni.isCurrentVersion() || ni.isPWC());
 | |
| 
 | |
|             // policies and relationships
 | |
|             info.setSupportsRelationships(true);
 | |
|             info.setSupportsPolicies(true);
 | |
| 
 | |
|             // renditions
 | |
|             info.setRenditionInfos(null);
 | |
|             List<RenditionData> renditions = object.getRenditions();
 | |
|             if (renditions != null && renditions.size() > 0)
 | |
|             {
 | |
|                 List<RenditionInfo> renditionInfos = new ArrayList<RenditionInfo>();
 | |
|                 for (RenditionData rendition : renditions)
 | |
|                 {
 | |
|                     RenditionInfoImpl renditionInfo = new RenditionInfoImpl();
 | |
|                     renditionInfo.setId(rendition.getStreamId());
 | |
|                     renditionInfo.setKind(rendition.getKind());
 | |
|                     renditionInfo.setContentType(rendition.getMimeType());
 | |
|                     renditionInfo.setTitle(rendition.getTitle());
 | |
|                     renditionInfo.setLength(rendition.getBigLength());
 | |
|                     renditionInfos.add(renditionInfo);
 | |
|                 }
 | |
|                 info.setRenditionInfos(renditionInfos);
 | |
|             }
 | |
| 
 | |
|             // relationships
 | |
|             setRelaionshipsToObjectInfo(object, info);
 | |
| 
 | |
|             // global settings
 | |
|             info.setHasAcl(true);
 | |
|             info.setSupportsDescendants(true);
 | |
|             info.setSupportsFolderTree(true);
 | |
|         }
 | |
| 
 | |
|         return info;
 | |
|     }
 | |
| 
 | |
|     private void setRelaionshipsToObjectInfo(ObjectData object, ObjectInfoImpl info)
 | |
|     {
 | |
|         info.setRelationshipSourceIds(null);
 | |
|         info.setRelationshipTargetIds(null);
 | |
| 
 | |
|         List<ObjectData> relationships = object.getRelationships();
 | |
|         if (relationships != null && relationships.size() > 0)
 | |
|         {
 | |
|             List<String> sourceIds = new ArrayList<String>();
 | |
|             List<String> targetIds = new ArrayList<String>();
 | |
|             for (ObjectData relationship : relationships)
 | |
|             {
 | |
|                 String sourceId = getIdProperty(relationship, PropertyIds.SOURCE_ID);
 | |
|                 String targetId = getIdProperty(relationship, PropertyIds.TARGET_ID);
 | |
|                 if (object.getId().equals(sourceId))
 | |
|                 {
 | |
|                     sourceIds.add(relationship.getId());
 | |
|                 }
 | |
|                 if (object.getId().equals(targetId))
 | |
|                 {
 | |
|                     targetIds.add(relationship.getId());
 | |
|                 }
 | |
|             }
 | |
|             if (sourceIds.size() > 0)
 | |
|             {
 | |
|                 info.setRelationshipSourceIds(sourceIds);
 | |
|             }
 | |
|             if (targetIds.size() > 0)
 | |
|             {
 | |
|                 info.setRelationshipTargetIds(targetIds);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // --------------------------------------------------------
 | |
| 
 | |
|     protected void checkRepositoryId(String repositoryId)
 | |
|     {
 | |
|         if (!connector.getRepositoryId().equals(repositoryId))
 | |
|         {
 | |
|             throw new CmisObjectNotFoundException("Unknown repository '" + repositoryId + "'!");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private Charset getEncoding(File tempFile, String mimeType)
 | |
|     {
 | |
|         Charset encoding = null;
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             InputStream tfis = new BufferedInputStream(new FileInputStream(tempFile));
 | |
|             ContentCharsetFinder charsetFinder = connector.getMimetypeService().getContentCharsetFinder();
 | |
|             encoding = charsetFinder.getCharset(tfis, mimeType);
 | |
|             tfis.close();
 | |
|         } catch (Exception e)
 | |
|         {
 | |
|             throw new CmisStorageException("Unable to read content: " + e.getMessage(), e);
 | |
|         }
 | |
| 
 | |
|         return encoding;
 | |
|     }
 | |
| 
 | |
|     private File copyToTempFile(ContentStream contentStream)
 | |
|     {
 | |
|         if (contentStream == null)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         File result = null;
 | |
|         try
 | |
|         {
 | |
|             result = TempFileProvider.createTempFile(contentStream.getStream(), "cmis", "content");
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             throw new CmisStorageException("Unable to store content: " + e.getMessage(), e);
 | |
|         }
 | |
| 
 | |
|         if ((contentStream.getLength() > -1) && (result == null || contentStream.getLength() != result.length()))
 | |
|         {
 | |
|             removeTempFile(result);
 | |
|             throw new CmisStorageException("Expected " + contentStream.getLength() + " bytes but retrieved " +
 | |
|                     (result == null ? -1 :result.length()) + " bytes!");
 | |
|         }
 | |
| 
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     private void removeTempFile(File tempFile)
 | |
|     {
 | |
|         if (tempFile == null)
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         try
 | |
|         {
 | |
|             tempFile.delete();
 | |
|         }
 | |
|         catch (Exception e)
 | |
|         {
 | |
|             // ignore - file will be removed by TempFileProvider
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void beforeCall()
 | |
|     {
 | |
|         AuthenticationUtil.pushAuthentication();
 | |
|         if (authentication != null)
 | |
|         {
 | |
|             // Use the previously-obtained authentication
 | |
|             AuthenticationUtil.setFullAuthentication(authentication);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (context == null)
 | |
|             {
 | |
|                 // Service not opened, yet
 | |
|                 return;
 | |
|             }
 | |
|             // Sticky sessions?
 | |
|             if (connector.openHttpSession())
 | |
|             {
 | |
|                 // create a session -> set a cookie
 | |
|                 // if the CMIS client supports cookies that might help in clustered environments
 | |
|                 ((HttpServletRequest) getContext().get(CallContext.HTTP_SERVLET_REQUEST)).getSession();
 | |
|             }
 | |
|             
 | |
|             // Authenticate
 | |
|             if (authentication != null)
 | |
|             {
 | |
|                 // We have already authenticated; just reuse the authentication
 | |
|                 AuthenticationUtil.setFullAuthentication(authentication);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // First check if we already are authenticated
 | |
|                 if (AuthenticationUtil.getFullyAuthenticatedUser() == null)
 | |
|                 {
 | |
|                     // We have to go to the repo and authenticate
 | |
|                     String user = context.getUsername();
 | |
|                     String password = context.getPassword();
 | |
|                     Authorization auth = new Authorization(user, password);
 | |
|                     if (auth.isTicket())
 | |
|                     {
 | |
|                         connector.getAuthenticationService().validate(auth.getTicket());
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         connector.getAuthenticationService().authenticate(auth.getUserName(), auth.getPasswordCharArray());
 | |
|                     }
 | |
|                 }
 | |
|                 this.authentication = AuthenticationUtil.getFullAuthentication();
 | |
|             }
 | |
|             
 | |
| //            // TODO: How is the proxy user working.
 | |
| //            //       Until we know what it is meant to do, it's not available
 | |
| //            String currentUser = connector.getAuthenticationService().getCurrentUserName();
 | |
| //            String user = getContext().getUsername();
 | |
| //            String password = getContext().getPassword();
 | |
| //            if (currentUser != null && currentUser.equals(connector.getProxyUser()))
 | |
| //            {
 | |
| //                if (user != null && user.length() > 0)
 | |
| //                {
 | |
| //                    AuthenticationUtil.setFullyAuthenticatedUser(user);
 | |
| //                }
 | |
| //            }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void afterCall()
 | |
|     {
 | |
|         AuthenticationUtil.popAuthentication();
 | |
|     }
 | |
| }
 |