mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-15 15:02:20 +00:00 
			
		
		
		
	43628: Fix for ALF-16299 - On full user profile, in the 'Sites' tab, the site descriptions are not aligned when a long site description is provided.
   43639: ALF-16701: use newly released Spring Surf 1.2.0-M1 rather than a SNAPSHOT
   43644: ALF-16527: webscript person.lib.ftl does not include all attributes for users out of the box.
       - added missing organizationId attribute.
   43649: ALF-16756: WebDAV: An error occurs on drag&drop content from local machine to alfresco when inbound move rule configured. 
   43651: ALF-16756: Fixed typos - I took this code in good faith!
   43659: ALF-16006: MT: Document Library is absent after upgrade from 3.4.x to 4.1.x (eg. 3.4.10 -> 4.1.1)
       - More patch dependencies required
   43666: ALF-16833 / MNT-187: Fixed regression in inbound rule firing caused by the fix to ALF-14744
   - Inbound rules were not firing for newly created content nodes that had null content properties (e.g. dataListItems)
   - Now the ALF-14744 fix has been revised so that inbound rules are only not fired for new nodes with zero-length content, as possibly created by the OSX / Windows 7 WebDAV clients during an upload, and only if policy.content.update.ignoreEmpty=true (the default)
   - OnContentUpdateRuleTrigger removed. Now all content property updates are handled by OnPropertyUpdateRuleTrigger so that we can have this subtle treatment of 'empty' content properties.
   - Reverted ALF-14744 changes to RuleServiceCoverageTest and added new tests for 'empty' content and ASPECT_NO_CONTENT
   - Updated RuleTriggerTest in line with changes
   43675: Merged V4.1 to V4.1-BUG-FIX
      43674: Merged PATCHES/V4.1.1 to V4.1
         43673: Merged V3.4-BUG-FIX to PATCHES/V4.1.1
            43672: ALF-16834, ALF-16833, MNT-187: Fix unit test failures and 'fine tune' logic to handle multiple updates within a single transaction creating a node
   43676: ALF-12324: Can't delete site with WQS content
   - Don't try to add nodes to the publish queue on site deletion
   43689: Fix for ALF-14207 - cm:link not correctly handled in Share's doclist when users don't have permission
   43690: Fix for ALF-16538 - Wrong label on change group permission message
   43696: ALF-16348 (Inconsistent error message when latest manager leaves a site)
   43719: ALF-16031: "CMISChangeLog does not log Created events"
   43720: ALF-14137: "When calling CMIS getAllVersions method using the OpenCMIS Browser the PreparedStatement is executed multiple times"
   43722: ALF-16352: Re-instated manage permissions action to document list view
   43745: Fix for ALF-15351.
   ContentModelFormProcessor had a TODO to handle associations defined on aspects other than
   those aspects already on the node for which a form is being submitted.
   I've added code that, when an association is added to a node, will check if the association
   is defined on any aspect in the system and if it is, it will let the association be created.
   43757: ALF-16171: If a password for admin user is specified with a space in the end during the installation Alfresco startup fails
      - Added validation that will not allow the specified password to contain leading to trailing white space
   43760: Fixed ALF-16317 "Labels not displaying full text when creating a rule on a folder with IE8" part 2
   43769: Fix for ALF-13461.
   Merged PATCHES/V3.4.8 to BRANCHES/DEV/V4.1-BUG-FIX:
      43344: MNT-162: Merged DEV to PATCHES/V3.4.8
           43254: MNT-162: CLONE - Hidden folder "Comments" is available to searching
              - Removed last '/' from COMMENT_QNAMEPATH constant.
   43770: ALF-15616: Merged V3.4-BUG-FIX (3.4.12) to V4.1-BUG-FIX (4.1.3)
      43768: Merged Dev to V3.4-BUG-FIX (3.4.12)
         43743: ALF-11956: WCM accessibility
         The problem with absence of initial focus on the first Web form element has been resolved in 'alfresco.xforms.XForm._loadHandler()'. This method creates all controls, defined in the form.
         'alfresco.xforms.FocusResolver._findControl()' has been renamed to a public-like method ('alfresco.xforms.FocusResolver.findControl()'). Also it has been modified to introduce a possibility of searching elements, using root XForms widgets container.
         Some other minor fixes.
   43772: Fixed ALF-16497 "Selected Items drop down box is expanded to the right in IE9"
   43787: Merged in Philippe's fix for ALF-16313:unmakeTranslation() called on pivot language gives exception
   43794: ALF-16155: "Notification digest accumulates and is repeatedly sent if 1 or more notification email fails"
   43805: ALF-16212: Fixes double encoding issue.
   43835: Fixed ALF-13595: Patches third party library date.js to prevent infinite loop.
   43836: 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).
   43844: ALF-16696: Merged DEV to V4.1-BUG-FIX
       43734: Share nodebrowser is unable to access node of a document with MS residual properties
           - Use localname if no prefix is registered for a namespace uri
   43864: Fixed ALF-16320 "Properties side panel collapses after editing properties in document preview window with IE8"
   43866: Fixed ALF-16320 "Properties side panel collapses after editing properties in document preview window with IE8" part 2
   - Checking using YAHOO.util.Event.getTarget instead
   43867: Fixed ALF-16276 "'"Web View' dahslet displays 2 scrollbars in IE8."
   43872: Merged V4.1 to V4.1-BUG-FIX
      43622: ALF-16757: Sharepoint doesn't work correct with SSO
      - Fix by Pavel
      43633: Latest translations from Gloria (r43623)
      43636: Merged PATCHES/V4.1.1 to V4.1
         43301: ALF-16811 / MNT-173: SOLR tracking spending too long evaluating paths
         - Too much time was being spent fetching the individual nodes in each path and there could be potentially thousands
         - Now we traverse all the ancestor parent associations in the cache before switching shared cache reads off, then bulk load them in one shot!
         43303: ALF-16812 / MNT-174: "dictionaryDAO.init() called, yet no namespace registry for domain" after node rejoins cluster
         - namespaceRegistryCache is secondary to the clustered dictionaryRegistryCache, so can be a non-clustered cache. Synchronization and thread locals already in use so still thread safe.
         - dictionaryDAO.init() now has sole responsibility of establishing the NamespaceRegistry threadlocal within its locks, so no more risk of cyclic dependencies, race conditions, partial initialization, or registries dropping out of the cache
         - To avoid being confused by the 'consistent read' behaviour of a transactional cache, DictionaryDAOImpl now reads / writes directly to a shared cache. Again locks and thread locals still used so still thread safe
         43334: ALF-16812 / MNT-174: Fixed failing unit tests
         - Because DictionaryDAOImpl now reads directly from the shared cache we need to reset it on initialization to avoid problems in multi-context unit tests
         43337: ALF-16811 / MNT-173: Fix test failure. Cope with IDs of deleted nodes in getCachedAncestors()
         43356: ALF-16811 / MNT-173: If we are disabling shared cache reads for the transaction, convert all existing reads and updates to avoid 'consistent read' behaviour giving us a potentially out of date node already accessed
         - Existing read buckets are simply thrown away
         - Updates are converted to removes to avoid any assumptions about existing shared cache content
         - New entries are left alone as they haven't come from the shared cache anyway
         43410: ALF-16813 / MNT-185: Web Scripts are being endlessly re-registering in clustered environment
         - Due to 'consistent read' behaviour of transactional cache
         - As accesses are regulated by RW locks we can read straight through to the shared cache instead
         43565: ALF-16814 / MNT-190: Bmlab Solr Node 2 threw unhandled NullPointerException (and possibly made solr unresponsive)
         - Added missing return statement
      43646: ALF-15755: Rationalization of WebDAVMethod.checkNode()
      43681: Merged HEAD to V4.1
         43656: Fix for ALF-16683 CMIS: cannot navigate to original document was created through CMIS with CHECKED OUT version state.
      43698: Latest Russian translations from Gloria
      43838: ALF-16875: Merged V4.1-BUG-FIX to V4.1
          43836: 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: ALF-16890 / 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).
      43868: Reverse merging r43838
      - Merge in wrong direction introducing duplicate fragment into share-documentlibrary-config.xml
      43871: ALF-16890: Merged PATCHES/V4.1.1 to V4.1
         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).
   43873: Merged V4.1 to V4.1-BUG-FIX (RECORD ONLY)
      43602: ALF-16254: Merged V4.1-BUG-FIX to V4.1
         43598: Merged HEAD to BRANCHES/DEV/V4.1-BUG-FIX
            41906: ALF-11378: REST API has been modified to return extra information about a user whether s/he belongs to a group or not.
      43612: ALF-16598: Merged V4.1-BUG-FIX to V4.1
          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.
          43254: 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
      43615: ALF-16794: Merged V4.1-BUG-FIX to V4.1
         43478: MNT-181: Now WebDAV will ALWAYS preserve the original metadata and versions of ANY node that is temporarily 'moved out' in ANY kind of 'shuffle' operation
         - To make the source node temporarily invisible to WebDAV the client specific HIDDEN aspect features are used
         - WebDAVHelper.isRenameShuffle() method introduced, to parallel ALF-3856 CIFS fix and using similar system.webdav.renameShufflePattern global property to detect the start of a shuffle
         - WebDAVHelper converted to use proper dependency injection
         - CopyMethod has become a simple subclass of MoveMethod as all the hidden aspect munging is done by it
         - DeleteMethod now preserves hidden nodes
         - PropFindMethod now ignores hidden nodes
         - Listing methods will hide hidden nodes from WebDAV
         43483: MNT-181: Corrected typo
         43523: MNT-181: Corrections
         - WebDAVLockService.unlock() made 'harmless' to call on already-unlocked nodes
         - Delete method hides rather than deletes versioned nodes and working copes in case it is called by OSX Finder during a 'replace' operation
         43524: MNT-181: Correction
         - PutMethod now 'unhides' hidden nodes and behaves as though it created them
         43570: MNT-181: More corrections researched by Valery
         - Don't treat all moves to temporary locations as copies - just those from non-temporary locations. Avoids initial upload leaving lots of hidden files around.
         - Only copy the content, not the whole node including aspects to avoid versioning temporary files!
         - Don't version on changes to sys:clientVisibilityMask - avoids 'double versioning'
         - Recognize Mac .TemporaryItems folder and ._ files as temporary
         43586: MNT-181: Final correction researched by Valery
         - Corrected system.webdav.renameShufflePattern so that it matches .TemporaryItems folder and ._ files as a full match
      43616: ALF-15755: Merged V4.1-BUG-FIX to V4.1    
          43591: ALF-16772: If the WebDAV path of a document exceeds 255 characters, documents opened in MSOffice cannot be saved back
              - Interpret null nodeLockToken as not locked.
      43629: Merged V4.1-BUG-FIX to V4.1 (4.1.2)
         43498: Fix for ALF-16648 - Alfresco Enterprise artifacts in artifacts.alfresco.com do not provide POM files / dependencies declarations:
         Merged HEAD to V4.1-BUG-FIX (4.1.2)
            43380: -- added site content for alfresco-platform-distribution POM
            43379: -- added site documentation for alfresco-platform-distribution POM
            43378: -- added site documentation for alfresco-platform-distribution POM
               -- deployed site for 4.2.b Community at https://artifacts.alfresco.com/nexus/content/repositories/alfresco-docs/alfresco-platform-distribution/latest/index.html
               -- created repository for Enterprise docs and added url in the appropriate edition properties
            43273: Use property to define POI version
            42966: ALF-14353 - Added platform distribution POM to standard maven-deploy procedure
            42965: ALF-14353 - added alfresco-platform-distribution to provide a Maven release descriptor (dependencyManagement) per each Community / Enterprise release
               -- moved maven-ant-tasks not to be in the runtime lib
               -- added platform distribution pom in the SDK folder
               -- updated maven.xml to deploy filter and deploy the appropriate platform-distribution POM per each releae
               -- in maven.xml moved configure-release and configure-snapshot goals to maven-env-prerequisites
               -- updated sdk readme to explain the presence of alfresco-platform-distribution POM
            42912: -- updated README header on the POM specifying it's NOT usable to build Alfresco
               -- make a clear reference to the POMs that get deployed by pom-experimental.xml being usable for development
            42842: ALF-14353: Fix artifactId alfresco-jlan -> alfresco-jlan-embed
            41883: ALF-14353 - fixed multiple Maven build issues. Now mvn clean install -f pom-experimental.xml works fine. Also deployed Spring Surf 1.2.0-SNAPSHOT so proper Surf version is retrieved
            41882: added pre-requisites to build POMs successfully with mvn clean install -f pom-experimental.xml
      43634: Merged V4.1-BUG-FIX to V4.1
         43386: ALF-13091: Prevent bean post processor propagation to child application contexts. Remove Jsr250BeanPostPorcessor from the CXF configuration, to prevent strange interaction with component scanning. 
      43641: Merged V4.1-BUG-FIX to V4.1 (4.1.2)
         43639: ALF-16701: use newly released Spring Surf 1.2.0-M1 in POM files rather than a SNAPSHOT
      43645: Merged V4.1-BUG-FIX to V4.1
          43644: ALF-16527: webscript person.lib.ftl does not include all attributes for users out of the box.
              - added missing organizationId attribute.
      43660: Merged V4.1-BUG-FIX to V4.1
          43659: ALF-16006: MT: Document Library is absent after upgrade from 3.4.x to 4.1.x (eg. 3.4.10 -> 4.1.1)
              - More patch dependencies required
      43669: Merged V4.1-BUG-FIX to V4.1
         43666: ALF-16833 / MNT-187: Fixed regression in inbound rule firing caused by the fix to ALF-14744
         - Inbound rules were not firing for newly created content nodes that had null content properties (e.g. dataListItems)
         - Now the ALF-14744 fix has been revised so that inbound rules are only not fired for new nodes with zero-length content, as possibly created by the OSX / Windows 7 WebDAV clients during an upload, and only if policy.content.update.ignoreEmpty=true (the default)
         - OnContentUpdateRuleTrigger removed. Now all content property updates are handled by OnPropertyUpdateRuleTrigger so that we can have this subtle treatment of 'empty' content properties.
         - Reverted ALF-14744 changes to RuleServiceCoverageTest and added new tests for 'empty' content and ASPECT_NO_CONTENT
         - Updated RuleTriggerTest in line with changes
      43697: Merged V4.1-BUG-FIX to V4.1
         43689: Fix for ALF-14207 - cm:link not correctly handled in Share's doclist when users don't have permission
      43761: Merged V4.1-BUG-FIX to V4.1
         43760: Fixed ALF-16317 "Labels not displaying full text when creating a rule on a folder with IE8" part 2
      43796: Merged V4.1-BUG-FIX to V4.1
         43795: Fix for ALF-16254 - "Leave Site" behaviour for group based site membership:
          Blah, Blah, Blah!
   43883: Fix for ALF-12711.
      Separated preparation of email message from sending of email message as described in
      analysis in JIRA. Did not merge the fix provided as MailActionExecuter.java.diff but
      reimplemented a fix based on that.
   43888: ALF-16781: Merged V3.4-BUG-FIX (3.4.12) to V4.1-BUG-FIX (4.1.3)
      43887: ALF-16898 CLONE - Transformation Server history shows incorrect transformation "To" type for images
         - ImageTransformActionExecuter ("Transform and Copy Image") re-factored to use ContentService
           rather than hard coded to ImageMagick
   43900: Merged V4.1 to V4.1-BUG-FIX
      43898: Merged PATCHES/V4.1.1 to V4.1
         43708: ALF-16903 / MNT-203: ACL changes not propagated between two non clustered Alfresco instances hitting same Solr and Database
         - Although SOLRTrackingComponentImpl was not trusting the cache when tracking node transactions, it was still trusting the cache for ACL change sets
         - This mean that when the SOLR node is using an Alfresco out of the cluster (which we are supposed to support) it could see stale ACLs
         - Fixed this by using the same strategy we use in AbstractNodeDAO
         - Now the main acl CRUD cache is set to ignore the shared cache during SOLR tracking
         - All secondary ACL caches are keyed by ACL ID AND version, so it prevents the possibility of retrieving a stale cached ACL for an old version
         43713: ALF-16903 / MNT-203: ACL changes not propagated between two non clustered Alfresco instances hitting same Solr and Database
         - Fix test failures
         - deleteAccessControlEntries must 'touch' all the ACLs it affects to keep caches valid
         - createAccessControlList should return the newly-versioned ACL
         43736: ALF-16904 / MNT-204: Index showing no progress on Solr server
         - SAP's tracking thread got stuck indefinitely waiting on a content response after sending a request, probably due to some misbehaving proxy or balancer
         - Now we make it possible to recover from this
         - A new alfresco.socketTimeout parameter is now supported in solrcore.properties
         - It specifies the number of milliseconds SOLR will wait before giving up waiting for data on an HTTP connection
         - The default is still zero which means wait indefinitely
         43759: ALF-16904 / MNT-204: Fixed compilation error
      43899: Merged PATCHES/V4.1.1 to V4.1 (RECORD ONLY)
         43667: Merged V4.1-BUG-FIX to PATCHES/V4.1.1
            43666: ALF-16833 / MNT-187: Fixed regression in inbound rule firing caused by the fix to ALF-14744
            - Inbound rules were not firing for newly created content nodes that had null content properties (e.g. dataListItems)
            - Now the ALF-14744 fix has been revised so that inbound rules are only not fired for new nodes with zero-length content, as possibly created by the OSX / Windows 7 WebDAV clients during an upload, and only if policy.content.update.ignoreEmpty=true (the default)
            - OnContentUpdateRuleTrigger removed. Now all content property updates are handled by OnPropertyUpdateRuleTrigger so that we can have this subtle treatment of 'empty' content properties.
            - Reverted ALF-14744 changes to RuleServiceCoverageTest and added new tests for 'empty' content and ASPECT_NO_CONTENT
            - Updated RuleTriggerTest in line with changes
   43901: Merged V3.4-BUG-FIX to V4.1-BUG-FIX
      43571: Merged DEV to V3.4-BUG-FIX 
       43569: ALF-16222: It's impossible to delete a file/message via IMAP using Microsoft Entourage 2008 in MacOSX 10.8
         fixed: UID failed.Existing file or folder error on attempt to delete file if deleted items already contains file with the same name.
         modified: AttachmentExtractor modified to avoid code dublication.
      43599: Fix for ALF-16505 - Discussion topics are sorted in ascending order (oldest first and newest last) when selecting 'All topics'
      43610: Merged Dev to V3.4-BUGFIX (3.4.12)
         43416: ALF-16470: SPP:Meeting recurent events are displayed incorrect in share calendar in all tabs (Day, Week, Month)
            Incorrect lucene query to search for events (it was search for events ONLY after fromDate, that gets from browser query. And recursive events, that have startDate before browser's query date, wasn't included into summary query for search for events.
            Start date is incorrect and is sets two times.
            Interval for month of end date is to small.
            Recursive event, that was started in previous month, and ends in current month, isn't included into result of search.
            For events, that ends on next date after start date, displays (view by month) only start date.
      43625: ALF-11817: Cope with incomplete lock token headers from Microsoft-WebDAV-MiniRedir without the enclosing <> by just consuming the whole string
      43670: ALF-11817: Prevent auto-hidden dot underscore files from reappearing on a put by checking for a shuffle path before 'unhiding'
      43746: Merged DEV to V3.4-BUG-FIX (with corrections)
         43692: ALF-16808 Webdav: Two versions of document have been added after the document has been rewritten once more via drag and drop action.
         1. Checked whether the current content property is empty.
         2. Disabled the versionable aspect.
         3. Added the new content to the node. 
      43763: Fix for ALF-14828 - Incorrect behavior on delete action (WCMQS site)
      43771: Fix for ALF-12752 - Custom form appearance parameters not accounted for causing ovverlapped textareas xforms changing their height
      43773: Merged DEV to V3.4-BUG-FIX (3.4.12)
         42010: ALF-14040: Event start/end time displays incorrect on MS Outlook Calendar and Calendar of created Meeting workspace
         1) Send a date for "Site Calendar" dashlet in ISO-8601 format (like in v4.1.1), then after transformation to client's time zone it is displayed correct
         2) For "My Calendar" dashlet we need to take into account that if the event is "all day event", then date should be used without time zone transformation
      43804: ALF-12326 HomeFolderProviderSynchronizer fails to move any user space which has a rule configured on it 
      43837: Fixes ALF-12145: Date.js patches merged to 3.4
      Merged BRANCHES/DEV/V4.0-BUG-FIX to BRANCHES/DEV/V3.4-BUG-FIX:
         36202: ALF-13483: Japanese: Incorrect date handle in a date Input filed
      Merged BRANCHES/DEV/V4.1-BUG-FIX to BRANCHES/DEV/V3.4-BUG-FIX:
         43835: Fixed ALF-13595: Patches third party library date.js to prevent infinite loop.
      43839: ALF-16869: Merged PATCHES/V3.4.8 to V3.4-BUG-FIX
          43344: MNT-162: Merged DEV to PATCHES/V3.4.8
              43254: MNT-162: CLONE - Hidden folder "Comments" is available to searching
                  - Removed last '/' from COMMENT_QNAMEPATH constant.
   43902: Merged V3.4-BUG-FIX to V4.1-BUG-FIX (RECORD ONLY)
      43177: Merged DEV to V3.4-BUG-FIX
         43087: ALF-16474: Records Management groups were not deleted after removing RM site
            - Delete Records Management groups on ASPECT_RECORDS_MANAGEMENT_ROOT delete
      	  - Backport of RM-190 from RM 2.0
      43228: ALF-16266: Merged HEAD to V3.4-BUG-FIX
         32846: Fixes: ALF-10519: Issues with translation of roles.
         31413: Fixes: ALF-10519 - Internationalises the role names for the repo browser's manage permissions page & makes these available to other pages through common.properties
      43229: ALF-16266: Fix  conflict data left in the properties file.
      43624: ALF-11817: Merged PATCHES/V4.0.2 to V3.4-BUG-FIX
         43587: Merged BRANCHES/V4.1-BUG-FIX to PATCHES/V4.0.2
            42363: ALF-16213: renaming versioned file results in file being deleted.
            43478: MNT-181: Now WebDAV will ALWAYS preserve the original metadata and versions of ANY node that is temporarily 'moved out' in ANY kind of 'shuffle' operation
            - To make the source node temporarily invisible to WebDAV the client specific HIDDEN aspect features are used
            - WebDAVHelper.isRenameShuffle() method introduced, to parallel ALF-3856 CIFS fix and using similar system.webdav.renameShufflePattern global property to detect the start of a shuffle
            - WebDAVHelper converted to use proper dependency injection
            - CopyMethod has become a simple subclass of MoveMethod as all the hidden aspect munging is done by it
            - DeleteMethod now preserves hidden nodes
            - PropFindMethod now ignores hidden nodes
            - Listing methods will hide hidden nodes from WebDAV
            43483: MNT-181: Corrected typo
            43523: MNT-181: Corrections
            - WebDAVLockService.unlock() made 'harmless' to call on already-unlocked nodes
            - Delete method hides rather than deletes versioned nodes and working copies in case it is called by OSX Finder during a 'replace' operation
            43524: MNT-181: Correction
            - PutMethod now 'unhides' hidden nodes and behaves as though it created them
            43570: MNT-181: More corrections researched by Valery
            - Don't treat all moves to temporary locations as copies - just those from non-temporary locations. Avoids initial upload leaving lots of hidden files around.
            - Only copy the content, not the whole node including aspects to avoid versioning temporary files!
            - Don't version on changes to sys:clientVisibilityMask - avoids 'double versioning'
            - Recognize Mac .TemporaryItems folder and ._ files as temporary
            43586: MNT-181: Final correction researched by Valery
            - Corrected system.webdav.renameShufflePattern so that it matches .TemporaryItems folder and ._ files as a full match
      43671: ALF-16834: Merged V4.1-BUG-FIX to V3.4-BUG-FIX
         43666: ALF-16833 / MNT-187: Fixed regression in inbound rule firing caused by the fix to ALF-14744
         - Inbound rules were not firing for newly created content nodes that had null content properties (e.g. dataListItems)
         - Now the ALF-14744 fix has been revised so that inbound rules are only not fired for new nodes with zero-length content, as possibly created by the OSX / Windows 7 WebDAV clients during an upload, and only if policy.content.update.ignoreEmpty=true (the default)
         - OnContentUpdateRuleTrigger removed. Now all content property updates are handled by OnPropertyUpdateRuleTrigger so that we can have this subtle treatment of 'empty' content properties.
         - Reverted ALF-14744 changes to RuleServiceCoverageTest and added new tests for 'empty' content and ASPECT_NO_CONTENT
         - Updated RuleTriggerTest in line with changes
      43842: Merged V4.0-BUG-FIX to V3.4-BUG-FIX
          33387: ALF-12492 - Email with empty subject sent to Alfresco by SMTP cause Null pointer Exception
      43843: ALF-16717: Merged V4.1-BUG-FIX to V3.4-BUG-FIX
          43314: ALF-16575 - Email server does not accept email where Subject ends with a period
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@43914 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			785 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			785 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2011 Alfresco Software Limited.
 | |
|  *
 | |
|  * This file is part of Alfresco
 | |
|  *
 | |
|  * Alfresco is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Lesser General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * Alfresco is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public License
 | |
|  * along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| package org.alfresco.repo.imap;
 | |
| 
 | |
| import java.io.IOException;
 | |
| import java.io.Serializable;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Collection;
 | |
| import java.util.Collections;
 | |
| import java.util.Date;
 | |
| import java.util.LinkedList;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| import java.util.NavigableMap;
 | |
| 
 | |
| import javax.mail.Flags;
 | |
| import javax.mail.Flags.Flag;
 | |
| import javax.mail.MessagingException;
 | |
| import javax.mail.internet.MimeMessage;
 | |
| 
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.model.ImapModel;
 | |
| import org.alfresco.repo.imap.AlfrescoImapConst.ImapViewMode;
 | |
| import org.alfresco.repo.imap.ImapService.FolderStatus;
 | |
| import org.alfresco.service.ServiceRegistry;
 | |
| import org.alfresco.service.cmr.model.FileExistsException;
 | |
| import org.alfresco.service.cmr.model.FileFolderService;
 | |
| import org.alfresco.service.cmr.model.FileInfo;
 | |
| import org.alfresco.service.cmr.model.FileNotFoundException;
 | |
| import org.alfresco.service.cmr.repository.NodeRef;
 | |
| import org.alfresco.service.cmr.security.AccessStatus;
 | |
| import org.alfresco.util.GUID;
 | |
| import org.alfresco.util.Utf7;
 | |
| import org.apache.commons.logging.Log;
 | |
| import org.apache.commons.logging.LogFactory;
 | |
| 
 | |
| import com.icegreen.greenmail.foedus.util.MsgRangeFilter;
 | |
| import com.icegreen.greenmail.store.FolderException;
 | |
| import com.icegreen.greenmail.store.FolderListener;
 | |
| import com.icegreen.greenmail.store.MailFolder;
 | |
| import com.icegreen.greenmail.store.MessageFlags;
 | |
| import com.icegreen.greenmail.store.SimpleStoredMessage;
 | |
| 
 | |
| /**
 | |
|  * Implementation of greenmail MailFolder. It represents an Alfresco content folder and handles
 | |
|  * appendMessage, copyMessage, expunge (delete), getMessages, getMessage and so requests.
 | |
|  * 
 | |
|  * @author Mike Shavnev
 | |
|  * @author David Ward
 | |
|  */
 | |
| public class AlfrescoImapFolder extends AbstractImapFolder implements Serializable
 | |
| {
 | |
|     private static final long serialVersionUID = -7223111284066976111L;
 | |
| 
 | |
|     private static Log logger = LogFactory.getLog(AlfrescoImapFolder.class);
 | |
| 
 | |
|     /**
 | |
|      * Reference to the {@link FileInfo} object representing the folder.
 | |
|      */
 | |
|     private final FileInfo folderInfo;
 | |
| 
 | |
|     /**
 | |
|      * Name of the folder.
 | |
|      */
 | |
|     private final String folderName;
 | |
| 
 | |
|     private final String folderPath;
 | |
|     
 | |
|     private final String userName;
 | |
| 
 | |
|     private final int mountPointId;
 | |
| 
 | |
|     /**
 | |
|      * Defines view mode.
 | |
|      */
 | |
|     private final ImapViewMode viewMode;
 | |
| 
 | |
|     /**
 | |
|      * Reference to the {@link ImapService} object.
 | |
|      */
 | |
|     private final ImapService imapService;
 | |
|     
 | |
|     /**
 | |
|      * Defines whether the folder is selectable or not.
 | |
|      */
 | |
|     private final boolean selectable;
 | |
| 
 | |
|     private final boolean extractAttachmentsEnabled;
 | |
|     
 | |
|     /**
 | |
|      * Cached Folder status information, validated against a change token.
 | |
|      */
 | |
|     private FolderStatus folderStatus;
 | |
|     
 | |
|     private static final Flags PERMANENT_FLAGS = new Flags();
 | |
| 
 | |
|     static
 | |
|     {
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.ANSWERED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.DELETED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.DRAFT);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.FLAGGED);
 | |
|         PERMANENT_FLAGS.add(Flags.Flag.SEEN);
 | |
|     }
 | |
|     
 | |
|     public boolean isExtractAttachmentsEnabled() 
 | |
|     {
 | |
|         return extractAttachmentsEnabled;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Protected constructor for the hierarchy delimiter
 | |
|      */
 | |
|     AlfrescoImapFolder(String userName, ImapService imapService, ServiceRegistry serviceRegistry)
 | |
|     {
 | |
|         this(null, userName, "", "", null, imapService, serviceRegistry, false, false, 0);
 | |
|     }
 | |
|         
 | |
| 
 | |
|     /**
 | |
|      * Constructs {@link AlfrescoImapFolder} object.
 | |
|      * 
 | |
|      * @param qualifiedMailboxName - name of the mailbox (e.g. "admin" for admin user).
 | |
|      * @param folderInfo - reference to the {@link FileInfo} object representing the folder.
 | |
|      * @param folderName - name of the folder.
 | |
|      * @param viewMode - defines view mode. Can be one of the following: {@link AlfrescoImapConst#MODE_ARCHIVE} or {@link AlfrescoImapConst#MODE_VIRTUAL}.
 | |
|      * @param rootNodeRef - reference to the root node of the store where folder is placed.
 | |
|      * @param mountPointName - name of the mount point.
 | |
|      */
 | |
|     public AlfrescoImapFolder(
 | |
|             FileInfo folderInfo,
 | |
|             String userName,
 | |
|             String folderName,
 | |
|             String folderPath,
 | |
|             ImapViewMode viewMode,
 | |
|             boolean extractAttachmentsEnabled,
 | |
|             ImapService imapService,
 | |
|             ServiceRegistry serviceRegistry,
 | |
|             int mountPointId)
 | |
|     {
 | |
|         this(folderInfo, userName, folderName, folderPath, viewMode, imapService, serviceRegistry, null, extractAttachmentsEnabled, mountPointId);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Constructs {@link AlfrescoImapFolder} object.
 | |
|      * 
 | |
|      * @param qualifiedMailboxName - name of the mailbox (e.g. "admin" for admin user).
 | |
|      * @param folderInfo - reference to the {@link FileInfo} object representing the folder.
 | |
|      * @param folderName - name of the folder.
 | |
|      * @param viewMode - defines view mode. Can be one of the following: {@link AlfrescoImapConst#MODE_ARCHIVE} or {@link AlfrescoImapConst#MODE_VIRTUAL}.
 | |
|      * @param rootNodeRef - reference to the root node of the store where folder is placed.
 | |
|      * @param mountPointName - name of the mount point.
 | |
|      * @param imapService - the IMAP service.
 | |
|      * @param selectable - defines whether the folder is selectable or not.
 | |
|      */
 | |
|     public AlfrescoImapFolder(
 | |
|             FileInfo folderInfo,
 | |
|             String userName,
 | |
|             String folderName,
 | |
|             String folderPath,
 | |
|             ImapViewMode viewMode,
 | |
|             ImapService imapService,
 | |
|             ServiceRegistry serviceRegistry,
 | |
|             Boolean selectable,
 | |
|             boolean extractAttachmentsEnabled,
 | |
|             int mountPointId)
 | |
|     {
 | |
|         super(serviceRegistry);
 | |
|         this.folderInfo = folderInfo;
 | |
|         this.userName = userName;
 | |
|         this.folderName = folderName != null ? folderName : (folderInfo != null ? folderInfo.getName() : null);
 | |
|         this.folderPath = folderPath;
 | |
|         this.viewMode = viewMode != null ? viewMode : ImapViewMode.ARCHIVE;
 | |
|         this.extractAttachmentsEnabled = extractAttachmentsEnabled;
 | |
|         this.imapService = imapService;
 | |
|         
 | |
|         // MailFolder object can be null if it is used to obtain hierarchy delimiter by LIST command:
 | |
|         // Example:
 | |
|         // C: 2 list "" ""
 | |
|         // S: * LIST () "." ""
 | |
|         // S: 2 OK LIST completed.
 | |
|         if (folderInfo != null)
 | |
|         {
 | |
|             if (selectable == null)
 | |
|             {
 | |
|                 // isSelectable();
 | |
|                 Boolean storedSelectable = !serviceRegistry.getNodeService().hasAspect(folderInfo.getNodeRef(), ImapModel.ASPECT_IMAP_FOLDER_NONSELECTABLE);
 | |
|                 if (storedSelectable == null)
 | |
|                 {
 | |
|                     this.selectable = true;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     this.selectable = storedSelectable;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 this.selectable = selectable;
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             this.selectable = false;
 | |
|         }
 | |
|         
 | |
|         this.mountPointId = mountPointId;
 | |
|     }
 | |
|     
 | |
|     /*
 | |
|      * (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#getFullName()
 | |
|      */
 | |
|     @Override
 | |
|     public String getFullName()
 | |
|     {
 | |
|         return Utf7.encode(AlfrescoImapConst.USER_NAMESPACE + AlfrescoImapConst.HIERARCHY_DELIMITER + this.userName
 | |
|                 + AlfrescoImapConst.HIERARCHY_DELIMITER + getFolderPath(), Utf7.UTF7_MODIFIED);
 | |
|     }
 | |
| 
 | |
|     /* (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#getName()
 | |
|      */
 | |
|     @Override
 | |
|     public String getName()
 | |
|     {
 | |
|         return this.folderName;
 | |
|     }
 | |
| 
 | |
|     /* (non-Javadoc)
 | |
|      * @see com.icegreen.greenmail.store.MailFolder#isSelectable()
 | |
|      */
 | |
|     @Override
 | |
|     public boolean isSelectable()
 | |
|     {
 | |
|         return this.selectable;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the contents of this folder.
 | |
|      * 
 | |
|      * @return A sorted map of UIDs to FileInfo objects.
 | |
|      */
 | |
|     private NavigableMap<Long, FileInfo> searchMails()
 | |
|     {
 | |
|         return getFolderStatus().search;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Invalidates the current cached state
 | |
|      * 
 | |
|      * @return <code>true</code> if this instance is still valid for reuse
 | |
|      */
 | |
|     public boolean reset()
 | |
|     {
 | |
|         this.folderStatus = null;
 | |
|         return new CommandCallback<Boolean>()
 | |
|         {
 | |
|             public Boolean command() throws Throwable
 | |
|             {
 | |
|                 return serviceRegistry.getNodeService().exists(folderInfo.getNodeRef());
 | |
|             }
 | |
|         }.run(true);
 | |
|     }
 | |
| 
 | |
|     protected FolderStatus getFolderStatus()
 | |
|     {
 | |
|         if (this.folderStatus == null)
 | |
|         {
 | |
|             CommandCallback<FolderStatus> command = new CommandCallback<FolderStatus>()
 | |
|             {
 | |
|                 public FolderStatus command() throws Throwable
 | |
|                 {
 | |
|                     return imapService.getFolderStatus(userName, folderInfo.getNodeRef(), viewMode);
 | |
|                 }
 | |
|             };
 | |
|             this.folderStatus = command.run();                         
 | |
|         }
 | |
|         return this.folderStatus;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Appends message to the folder.
 | |
|      * 
 | |
|      * @param message - message.
 | |
|      * @param flags - message flags.
 | |
|      * @param internalDate - not used. Current date used instead.
 | |
|      */
 | |
|     @Override
 | |
|     protected long appendMessageInternal(
 | |
|             MimeMessage message,
 | |
|             Flags flags,
 | |
|             Date internalDate)
 | |
|             throws FileExistsException, FileNotFoundException, IOException, MessagingException 
 | |
|     {
 | |
|         long uid = createMimeMessageInFolder(this.folderInfo, message, flags);
 | |
|         // Invalidate current folder status
 | |
|         this.folderStatus = null;
 | |
|         return uid;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Copies message with the given UID to the specified {@link MailFolder}.
 | |
|      * 
 | |
|      * @param uid - UID of the message
 | |
|      * @param toFolder - reference to the destination folder.
 | |
|      * @throws MessagingException 
 | |
|      * @throws IOException 
 | |
|      * @throws FileNotFoundException 
 | |
|      * @throws FileExistsException 
 | |
|      */
 | |
|     @Override
 | |
|     protected void copyMessageInternal(
 | |
|             long uid, MailFolder toFolder)
 | |
|             throws MessagingException, FileExistsException, FileNotFoundException, IOException 
 | |
|     {
 | |
|         AlfrescoImapFolder toImapMailFolder = (AlfrescoImapFolder) toFolder;
 | |
| 
 | |
|         NodeRef destFolderNodeRef = toImapMailFolder.getFolderInfo().getNodeRef();
 | |
| 
 | |
|         FileInfo sourceMessageFileInfo = searchMails().get(uid);
 | |
| 
 | |
|         if (serviceRegistry.getNodeService().hasAspect(sourceMessageFileInfo.getNodeRef(), ImapModel.ASPECT_IMAP_CONTENT))
 | |
|         {
 | |
|                 //Generate body of message
 | |
|             MimeMessage newMessage = new ImapModelMessage(sourceMessageFileInfo, serviceRegistry, true);
 | |
|             toImapMailFolder.appendMessageInternal(newMessage, imapService.getFlags(sourceMessageFileInfo), new Date());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             String fileName = (String) serviceRegistry.getNodeService().getProperty(sourceMessageFileInfo.getNodeRef(), ContentModel.PROP_NAME);
 | |
|             String newFileName = imapService.generateUniqueFilename(destFolderNodeRef, fileName);
 | |
|             serviceRegistry.getFileFolderService().copy(sourceMessageFileInfo.getNodeRef(), destFolderNodeRef, newFileName);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Marks all messages in the folder as deleted using {@link Flags.Flag#DELETED} flag.
 | |
|      */
 | |
|     @Override
 | |
|     public void deleteAllMessagesInternal() throws FolderException
 | |
|     {
 | |
|         if (isReadOnly())
 | |
|         {
 | |
|             throw new FolderException("Can't delete all - Permission denied");
 | |
|         }
 | |
|         
 | |
|         for (Map.Entry<Long, FileInfo> entry : searchMails().entrySet())
 | |
|         {
 | |
|             imapService.setFlag(entry.getValue(), Flags.Flag.DELETED, true);
 | |
|             // comment out to physically remove content.
 | |
|             // fileFolderService.delete(fileInfo.getNodeRef());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Deletes messages marked with {@link Flags.Flag#DELETED}. Note that this message deletes all messages with this flag.
 | |
|      */
 | |
|     @Override
 | |
|     protected void expungeInternal() throws FolderException
 | |
|     {
 | |
|         if (isReadOnly())
 | |
|         {
 | |
|             throw new FolderException("Can't expunge - Permission denied");
 | |
|         }
 | |
| 
 | |
|         for (Map.Entry<Long, FileInfo> entry : searchMails().entrySet())
 | |
|         {
 | |
|             imapService.expungeMessage(entry.getValue());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the MSN number of the first unseen message.
 | |
|      * 
 | |
|      * @return MSN number of the first unseen message.
 | |
|      */
 | |
|     @Override
 | |
|     public int getFirstUnseen()
 | |
|     {
 | |
|         return getFolderStatus().firstUnseen;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns message by its UID.
 | |
|      * 
 | |
|      * @param uid - UID of the message.
 | |
|      * @return message.
 | |
|      * @throws MessagingException 
 | |
|      */
 | |
|     @Override
 | |
|     protected SimpleStoredMessage getMessageInternal(long uid) throws MessagingException
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessageInternal] " + this);
 | |
|         }
 | |
|         FileInfo mesInfo = searchMails().get(uid); 
 | |
|         if (mesInfo == null)
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|         return imapService.getMessage(mesInfo);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of the messages in the folder.
 | |
|      * 
 | |
|      * @return Count of the messages.
 | |
|      */
 | |
|     @Override
 | |
|     public int getMessageCount()
 | |
|     {
 | |
|         return getFolderStatus().messageCount;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns UIDs of all messages in the folder.
 | |
|      * 
 | |
|      * @return UIDS of the messages.
 | |
|      */
 | |
|     @Override
 | |
|     public long[] getMessageUids()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessageUidsInternal] " + this);
 | |
|         }
 | |
|         
 | |
|         Collection<Long> uidSet = searchMails().keySet();
 | |
|         long[] uids = new long[uidSet.size()];
 | |
|         int i = 0;
 | |
|         for (Long uid : uidSet)
 | |
|         {
 | |
|             uids[i++] = uid;
 | |
|         }
 | |
|         return uids;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns list of all messages in the folder.
 | |
|      * 
 | |
|      * @return list of {@link SimpleStoredMessage} objects.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getMessagesInternal()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getMessagesInternal] " + this);
 | |
|         }
 | |
|         return convertToMessages(searchMails().values());
 | |
|     }
 | |
| 
 | |
|     private List<SimpleStoredMessage> convertToMessages(Collection<FileInfo> fileInfos)
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[convertToMessages] " + this);
 | |
|         }
 | |
|         if (fileInfos == null || fileInfos.size() == 0)
 | |
|         {
 | |
|             logger.debug("[convertToMessages] - fileInfos is empty or null");
 | |
|             return Collections.emptyList();
 | |
|         }
 | |
|         List<SimpleStoredMessage> result = new LinkedList<SimpleStoredMessage>();
 | |
|         for (FileInfo fileInfo : fileInfos)
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 result.add(imapService.createImapMessage(fileInfo, false));
 | |
|                 if (logger.isDebugEnabled())
 | |
|                 {
 | |
|                     logger.debug("[convertToMessages] Message added: " + fileInfo.getName());
 | |
|                 }
 | |
|             }
 | |
|             catch (MessagingException e)
 | |
|             {
 | |
|                 logger.warn("[convertToMessages] Invalid message! File name:" + fileInfo.getName(), e);
 | |
|             }
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns list of messages by filter.
 | |
|      * 
 | |
|      * @param msgRangeFilter - {@link MsgRangeFilter} object representing filter.
 | |
|      * @return list of filtered messages.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getMessagesInternal(MsgRangeFilter msgRangeFilter)
 | |
|     {
 | |
|         throw new UnsupportedOperationException("IMAP implementation doesn't support POP3 requests");
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns message sequence number in the folder by its UID.
 | |
|      * 
 | |
|      * @param uid - message UID.
 | |
|      * @return message sequence number.
 | |
|      * @throws FolderException if no message with given UID.
 | |
|      */
 | |
|     @Override
 | |
|     public int getMsn(long uid) throws FolderException
 | |
|     {
 | |
|         NavigableMap<Long, FileInfo> messages = searchMails();
 | |
|         if (!messages.containsKey(uid))
 | |
|         {
 | |
|             throw new FolderException("No such message.");            
 | |
|         }
 | |
|         return messages.headMap(uid, true).size();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the list of messages that have no {@link Flags.Flag#DELETED} flag set for current user.
 | |
|      * 
 | |
|      * @return the list of non-deleted messages.
 | |
|      */
 | |
|     @Override
 | |
|     protected List<SimpleStoredMessage> getNonDeletedMessagesInternal()
 | |
|     {
 | |
|         if (logger.isDebugEnabled())
 | |
|         {
 | |
|             logger.debug("[getNonDeletedMessagesInternal] " + this);
 | |
|         }
 | |
|         List<SimpleStoredMessage> result = new ArrayList<SimpleStoredMessage>();
 | |
| 
 | |
|         Collection<SimpleStoredMessage> values = getMessagesInternal();
 | |
|         for (SimpleStoredMessage message : values)
 | |
|         {
 | |
|             if (!getFlags(message).contains(Flags.Flag.DELETED))
 | |
|             {
 | |
|                 result.add(message);
 | |
|             }
 | |
| 
 | |
|         }
 | |
|         if (logger.isDebugEnabled() && folderInfo != null)
 | |
|         {
 | |
|             logger.debug(folderInfo.getName() + " - Non deleted messages count:" + result.size());
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns permanent flags.
 | |
|      * 
 | |
|      * @return {@link Flags} object containing flags.
 | |
|      */
 | |
|     @Override
 | |
|     public Flags getPermanentFlags()
 | |
|     {
 | |
|         return PERMANENT_FLAGS;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of messages with {@link Flags.Flag#RECENT} flag.
 | |
|      * If {@code reset} parameter is {@code true} - removes {@link Flags.Flag#RECENT} flag from
 | |
|      * the message for current user.
 | |
|      * 
 | |
|      * @param reset - if true the {@link Flags.Flag#RECENT} will be deleted for current user if exists.
 | |
|      * @return returns count of recent messages.
 | |
|      */
 | |
|     @Override
 | |
|     public int getRecentCount(boolean reset)
 | |
|     {
 | |
|         int recent = getFolderStatus().recentCount;
 | |
|         if (reset && recent > 0)
 | |
|         {
 | |
|             CommandCallback<Void> command = new CommandCallback<Void>()
 | |
|             {
 | |
|                 public Void command() throws Throwable
 | |
|                 {
 | |
|                     for (FileInfo fileInfo : folderStatus.search.values())
 | |
|                     {
 | |
|                         Flags flags = imapService.getFlags(fileInfo);
 | |
|                         if (flags.contains(Flags.Flag.RECENT))
 | |
|                         {
 | |
|                             imapService.setFlag(fileInfo, Flags.Flag.RECENT, false);
 | |
|                         }
 | |
|                     }
 | |
|                     return null;
 | |
|                 }
 | |
|             };
 | |
|             command.run();
 | |
|         }
 | |
|         return recent;        
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns UIDNEXT value of the folder.
 | |
|      * 
 | |
|      * @return UIDNEXT value.
 | |
|      */
 | |
|     @Override
 | |
|     public long getUidNext()
 | |
|     {
 | |
|         NavigableMap<Long, FileInfo> search = getFolderStatus().search; 
 | |
|         return search.isEmpty() ? 1 : search.lastKey() + 1;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Returns UIDVALIDITY value of the folder.
 | |
|      * 
 | |
|      * @return UIDVALIDITY value.
 | |
|      */
 | |
|     @Override
 | |
|     public long getUidValidity()
 | |
|     {
 | |
|         return getFolderStatus().uidValidity + mountPointId;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns count of the messages with {@link Flags.Flag#SEEN} in the folder for the current user.
 | |
|      * 
 | |
|      * @return Count of the unseen messages for current user.
 | |
|      */
 | |
|     @Override
 | |
|     public int getUnseenCount()
 | |
|     {
 | |
|         return getFolderStatus().unseenCount;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Replaces flags for the message with the given UID. If {@code addUid} is set to {@code true}
 | |
|      * {@link FolderListener} objects defined for this folder will be notified.
 | |
|      * {@code silentListener} can be provided - this listener wouldn't be notified.
 | |
|      * 
 | |
|      * @param flags - new flags.
 | |
|      * @param uid - message UID.
 | |
|      * @param silentListener - listener that shouldn't be notified.
 | |
|      * @param addUid - defines whether or not listeners be notified.
 | |
|      * @throws FolderException 
 | |
|      * @throws MessagingException 
 | |
|      */
 | |
|     @Override
 | |
|     protected void replaceFlagsInternal(
 | |
|             Flags flags,
 | |
|             long uid,
 | |
|             FolderListener silentListener,
 | |
|             boolean addUid)
 | |
|             throws FolderException, MessagingException 
 | |
|     {
 | |
|         int msn = getMsn(uid);
 | |
|         FileInfo fileInfo = searchMails().get(uid);
 | |
|         imapService.setFlags(fileInfo, MessageFlags.ALL_FLAGS, false);
 | |
|         imapService.setFlags(fileInfo, flags, true);
 | |
|         
 | |
|         Long uidNotification = addUid ? uid : null;
 | |
|         notifyFlagUpdate(msn, flags, uidNotification, silentListener);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets flags for the message with the given UID. If {@code addUid} is set to {@code true}
 | |
|      * {@link FolderListener} objects defined for this folder will be notified.
 | |
|      * {@code silentListener} can be provided - this listener wouldn't be notified.
 | |
|      * 
 | |
|      * @param flags - new flags.
 | |
|      * @param value - flags value.
 | |
|      * @param uid - message UID.
 | |
|      * @param silentListener - listener that shouldn't be notified.
 | |
|      * @param addUid - defines whether or not listeners be notified.
 | |
|      * @throws MessagingException 
 | |
|      * @throws FolderException 
 | |
|      */
 | |
|     @Override
 | |
|     protected void setFlagsInternal(
 | |
|             Flags flags,
 | |
|             boolean value,
 | |
|             long uid,
 | |
|             FolderListener silentListener,
 | |
|             boolean addUid)
 | |
|             throws MessagingException, FolderException 
 | |
|     {
 | |
|         int msn = getMsn(uid);
 | |
|         FileInfo fileInfo = searchMails().get(uid);
 | |
|         imapService.setFlags(fileInfo, flags, value);
 | |
|         
 | |
|         Long uidNotification = null;
 | |
|         if (addUid)
 | |
|         {
 | |
|             uidNotification = new Long(uid);
 | |
|         }
 | |
|         notifyFlagUpdate(msn, flags, uidNotification, silentListener);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private Flags getFlags(SimpleStoredMessage mess)
 | |
|     {
 | |
|         return ((AbstractMimeMessage) mess.getMimeMessage()).getFlags();
 | |
|     }
 | |
| 
 | |
|     // ----------------------Getters and Setters----------------------------
 | |
| 
 | |
|     public String getFolderPath()
 | |
|     {
 | |
|         return this.folderPath;
 | |
|     }
 | |
| 
 | |
|     public FileInfo getFolderInfo()
 | |
|     {
 | |
|         return folderInfo;
 | |
|     }
 | |
|     
 | |
|     /* (non-Javadoc)
 | |
|      * @see org.alfresco.repo.imap.AbstractImapFolder#isMarkedInternal()
 | |
|      */
 | |
|     @Override
 | |
|     public boolean isMarked()
 | |
|     {
 | |
|         FolderStatus folderStatus = getFolderStatus();
 | |
|         return folderStatus.recentCount > 0 || folderStatus.unseenCount > 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Whether the folder is read-only for user.
 | |
|      * 
 | |
|      * @return {@code boolean}
 | |
|      */
 | |
|     @Override
 | |
|     protected boolean isReadOnly()
 | |
|     {
 | |
|         AccessStatus status = serviceRegistry.getPublicServiceAccessService().hasAccess(ServiceRegistry.NODE_SERVICE.getLocalName(), "createNode", folderInfo.getNodeRef(), null, null, null);
 | |
|         //serviceRegistry.getPermissionService().hasPermission(folderInfo.getNodeRef(), PermissionService.WRITE);
 | |
|         return  status == AccessStatus.DENIED;
 | |
|     }
 | |
| 
 | |
|     public ImapViewMode getViewMode()
 | |
|     {
 | |
|         return viewMode;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      *  Creates the EML message in the specified folder.
 | |
|      *  
 | |
|      *  @param folderFileInfo The folder to create message in.
 | |
|      *  @param message The original MimeMessage.
 | |
|      *  @return ID of the new message created 
 | |
|      * @throws FileNotFoundException 
 | |
|      * @throws FileExistsException 
 | |
|      * @throws MessagingException 
 | |
|      * @throws IOException 
 | |
|      */
 | |
|     private long createMimeMessageInFolder(
 | |
|             FileInfo folderFileInfo,
 | |
|             MimeMessage message,
 | |
|             Flags flags)
 | |
|             throws FileExistsException, FileNotFoundException, IOException, MessagingException 
 | |
|     {
 | |
|         String name = AlfrescoImapConst.MESSAGE_PREFIX + GUID.generate();
 | |
|         FileFolderService fileFolderService = serviceRegistry.getFileFolderService();
 | |
|         FileInfo messageFile = fileFolderService.create(folderFileInfo.getNodeRef(), name, ContentModel.TYPE_CONTENT);
 | |
|         final long newMessageUid = (Long) messageFile.getProperties().get(ContentModel.PROP_NODE_DBID);
 | |
|         name = AlfrescoImapConst.MESSAGE_PREFIX  + newMessageUid + AlfrescoImapConst.EML_EXTENSION;
 | |
|         fileFolderService.rename(messageFile.getNodeRef(), name);
 | |
|         Flags newFlags = new Flags(flags);
 | |
|         newFlags.add(Flag.RECENT);
 | |
|         imapService.setFlags(messageFile, newFlags, true);
 | |
|         
 | |
|         if (extractAttachmentsEnabled)
 | |
|         {
 | |
|             imapService.extractAttachments(messageFile.getNodeRef(), message);
 | |
|         }
 | |
|         // Force persistence of the message to the repository
 | |
|         new IncomingImapMessage(messageFile, serviceRegistry, message);
 | |
|         return newMessageUid;        
 | |
|     }
 | |
| }
 |