mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-22 15:12:38 +00:00 
			
		
		
		
	15128: Merge 3.1 to 3.2:
   		15114: Added support for impersonation level sharing mode check, to fix Office2007 file open issue. ETHREEOH-2320.
	15130: Record-only
	15340: Merge 3.1 to 3.2:
   		14359: Fixed native call being used when <disableNative/> was configured. ETHREEOH-2105. (Record-only)
   		14484: Merged HEAD to v3.1: (Record-only)
       	13943 Added FTP IPv6 configuration, via the <IPv6 state="enabled|disabled"/> tag. Added the ftp.ipv6 property. MOB-714.
   		14523: Add trailing 'A' to CIFS server name, removed by recent checkin. (Record-only)
   		14561: Change the file server config bean to use the 'org.alfresco.fileserver' logging level.
   		14916: Fixes for local domain lookup when WINS is configured. ETHREEOH-2263.
   		14922: Merge HEAD to V3.1
       	14626: Fixes for the client side Windows desktop action application. part of ETHREEOH-401
   		15155: Fixes to client side desktop action exe handling of paths that are not mapped to the root of the Alfresco share. ETHREEOH-1613
	15341: Record-only
	15549: Check for null ClientInfo in the setCurrentUser() method and clear the auth context. Part of ETHREEOH-2538.
	15550: Fixed performance issue in the continue search code, add warn level output of folder search timing.
	15564: Merge 3.1 to 3.2:
   		14964: Port fix for convert content I/O exceptions to file server exceptions during write and truncate (part 2). ETWOTWO-1241
   		15233: Ignore nodes that no longer exist during the second stage of a file server folder search.
   		15234: Fixed incorrect length check when buffering MSOffice document writes.
	15565: Record-only
	15568: Fix for cut/paste file between folders on CIFS. ETHREEOH-2323 + ENH-515.
	15569: Record-only
	15644: Changed filesystem debug setting so it works with old and new config styles.
	15786: Record-only
	15787: Port of repo filesystem MS Office document locking fix. ETHREEOH-2579
git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@16122 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
	
		
			
				
	
	
		
			875 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			875 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2006 Alfresco, Inc.
 | |
|  *
 | |
|  * Licensed under the Mozilla Public License version 1.1 
 | |
|  * with a permitted attribution clause. You may obtain a
 | |
|  * copy of the License at
 | |
|  *
 | |
|  *   http://www.alfresco.org/legal/license.txt
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing,
 | |
|  * software distributed under the License is distributed on an
 | |
|  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
 | |
|  * either express or implied. See the License for the specific
 | |
|  * language governing permissions and limitations under the
 | |
|  * License.
 | |
|  */
 | |
| 
 | |
| #include "stdafx.h"
 | |
| #include "CAlfrescoApp.h"
 | |
| #include "CAlfrescoAppDlg.h"
 | |
| 
 | |
| #include <stdlib.h>
 | |
| 
 | |
| #include "util\String.h"
 | |
| #include "util\DataBuffer.h"
 | |
| #include "util\FileName.h"
 | |
| #include "util\Integer.h"
 | |
| #include "util\Debug.h"
 | |
| 
 | |
| #include <shellapi.h>
 | |
| 
 | |
| #ifdef _DEBUG
 | |
| #define new DEBUG_NEW
 | |
| #endif
 | |
| 
 | |
| using namespace std;
 | |
| using namespace Alfresco;
 | |
| 
 | |
| // CCAlfrescoAppApp
 | |
| 
 | |
| BEGIN_MESSAGE_MAP(CAlfrescoApp, CWinApp)
 | |
| 	ON_COMMAND(ID_HELP, CWinApp::OnHelp)
 | |
| END_MESSAGE_MAP()
 | |
| 
 | |
| // CCAlfrescoApp construction
 | |
| 
 | |
| CAlfrescoApp::CAlfrescoApp()
 | |
| {
 | |
| 	// TODO: add construction code here,
 | |
| 	// Place all significant initialization in InitInstance
 | |
| }
 | |
| 
 | |
| 
 | |
| // The one and only CCAlfrescoAppApp object
 | |
| 
 | |
| CAlfrescoApp theApp;
 | |
| 
 | |
| 
 | |
| // CCAlfrescoAppApp initialization
 | |
| 
 | |
| BOOL CAlfrescoApp::InitInstance()
 | |
| {
 | |
| 	// InitCommonControls() is required on Windows XP if an application
 | |
| 	// manifest specifies use of ComCtl32.dll version 6 or later to enable
 | |
| 	// visual styles.  Otherwise, any window creation will fail.
 | |
| 
 | |
| 	InitCommonControls();
 | |
| 	CWinApp::InitInstance();
 | |
| 	AfxEnableControlContainer();
 | |
| 
 | |
| 	// Check if debug logging is enabled
 | |
| 
 | |
| 	char dbgLogName[MAX_PATH];
 | |
| 	size_t dbgLogSize;
 | |
| 
 | |
| 	if ( getenv_s( &dbgLogSize, dbgLogName, sizeof( dbgLogName), "ALFDEBUG") == 0) {
 | |
| 
 | |
| 		// Enable debug output
 | |
| 
 | |
| 		Debug::openLog( dbgLogName);
 | |
| 
 | |
| 		// Log the application startup
 | |
| 
 | |
| 		DBGOUT_TS << "---------- Desktop client app started ----------" << endl; 
 | |
| 	}
 | |
| 
 | |
| 	// Get the application path
 | |
| 
 | |
| 	String appPath = __wargv[0];
 | |
| 
 | |
| 	int pos = appPath.lastIndexOf(PathSeperator);
 | |
| 
 | |
| 	if ( pos < 0) {
 | |
| 		AfxMessageBox( L"Invalid application path", MB_OK | MB_ICONSTOP);
 | |
| 		DBGOUT_TS << "Error, bad application path, " << appPath << endl;
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	// Get the path to the folder containing the application
 | |
| 
 | |
| 	String folderPath = appPath.substring(0, pos);
 | |
| 	String exeName    = appPath.substring(pos + 1);
 | |
| 
 | |
| 	// Create a list of the command line arguments
 | |
| 
 | |
| 	StringList argList;
 | |
| 	bool argSetWorkDir = false;
 | |
| 
 | |
| 	for ( int i = 1; i < __argc; i++) {
 | |
| 
 | |
| 		// Check if the argument is a path or switch
 | |
| 
 | |
| 		String arg = __wargv[i];
 | |
| 
 | |
| 		if ( arg.startsWith( "/")) {
 | |
| 
 | |
| 			// Check for the set working directory switch
 | |
| 
 | |
| 			if ( arg.equalsIgnoreCase( "/D")) {
 | |
| 				argSetWorkDir = true;
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "/D switch specified" << endl;
 | |
| 			}
 | |
| 			else {
 | |
| 				String msg = L"Invalid command line switch - ";
 | |
| 				msg.append( arg);
 | |
| 				AfxMessageBox( msg.data(), MB_OK | MB_ICONSTOP);
 | |
| 				DBGOUT_TS << "Error, " << msg << endl;
 | |
| 				return 2;
 | |
| 			}
 | |
| 		}
 | |
| 		else {
 | |
| 
 | |
| 			// Add the path to the argument list
 | |
| 
 | |
| 			argList.addString( arg);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Check if the working directory should be set to the path of the first document
 | |
| 
 | |
| 	if ( argSetWorkDir == true) {
 | |
| 
 | |
| 		// Check if there are any document paths
 | |
| 
 | |
| 		if ( argList.numberOfStrings() == 0) {
 | |
| 			AfxMessageBox( L"Cannot set working directory, no document paths", MB_OK | MB_ICONSTOP);
 | |
| 			DBGOUT_TS << "Error, cannot set working directory, no document paths" << endl;
 | |
| 			return 3;
 | |
| 		}
 | |
| 
 | |
| 		// Get the first document path and remove the file name
 | |
| 
 | |
| 		String docPath = argList[0];
 | |
| 		pos = docPath.lastIndexOf( PathSeperator);
 | |
| 
 | |
| 		if ( pos < 0) {
 | |
| 			AfxMessageBox( L"Invalid document path", MB_OK | MB_ICONSTOP);
 | |
| 			DBGOUT_TS << "Error, invalid document path, " << docPath << endl;
 | |
| 			return 4;
 | |
| 		}
 | |
| 
 | |
| 		// Set the document path as the working directory folder
 | |
| 
 | |
| 		folderPath = docPath.substring(0, pos);
 | |
| 
 | |
| 		// DEBUG
 | |
| 
 | |
| 		DBGOUT_TS << "Using document path as working directory, " << folderPath << endl;
 | |
| 	}
 | |
| 
 | |
| 	// DEBUG
 | |
| 
 | |
| 	if ( HAS_DEBUG)
 | |
| 		DBGOUT_TS << "Using folder path " << folderPath << " for Alfresco base dir" << endl;
 | |
| 
 | |
| 	// Create the Alfresco interface
 | |
| 
 | |
| 	AlfrescoInterface alfresco(folderPath);
 | |
| 
 | |
| 	if ( alfresco.isAlfrescoFolder()) {
 | |
| 
 | |
| 		try {
 | |
| 
 | |
| 			// DEBUG
 | |
| 
 | |
| 			DBGOUT_TS << "Using folder " << folderPath << endl;
 | |
| 
 | |
| 			// Get the action information
 | |
| 
 | |
| 			AlfrescoActionInfo actionInfo = alfresco.getActionInformation(exeName);
 | |
| 
 | |
| 			// DEBUG
 | |
| 
 | |
| 			if ( HAS_DEBUG) {
 | |
| 				DBGOUT_TS << "Action " << actionInfo.getName() << endl;
 | |
| 				DBGOUT_TS << "  PreProcess: ";
 | |
| 
 | |
| 				if ( actionInfo.hasPreProcessAction( PreConfirmAction))
 | |
| 					DBGOUT << "Confirm ";
 | |
| 				if ( actionInfo.hasPreProcessAction( PreCopyToTarget))
 | |
| 					DBGOUT << "CopyToTarget ";
 | |
| 				if ( actionInfo.hasPreProcessAction( PreLocalToWorkingCopy))
 | |
| 					DBGOUT << "LocalToWorkingCopy";
 | |
| 				DBGOUT << endl;
 | |
| 			}
 | |
| 
 | |
| 			// Check if the action should be confirmed
 | |
| 
 | |
| 			if ( actionInfo.hasPreProcessAction(PreConfirmAction)) {
 | |
| 
 | |
| 				// Get the confirmation message
 | |
| 
 | |
| 				String confirmMsg = actionInfo.getConfirmationMessage();
 | |
| 				if ( confirmMsg.length() == 0)
 | |
| 					confirmMsg = L"Run action ?";
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "Confirm action, message = " << confirmMsg << endl;
 | |
| 
 | |
| 				// Display a confirmation dialog
 | |
| 
 | |
| 				if ( AfxMessageBox( confirmMsg, MB_OKCANCEL | MB_ICONQUESTION) == IDCANCEL) {
 | |
| 					DBGOUT_TS << "User cancelled action" << endl;
 | |
| 					return FALSE;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// Check if the action supports multiple paths, if not then call the action once for each supplied path
 | |
| 
 | |
| 			if ( actionInfo.hasAttribute(AttrMultiplePaths)) {
 | |
| 
 | |
| 				// Run the action
 | |
| 
 | |
| 				runAction( alfresco, argList, actionInfo);
 | |
| 			}
 | |
| 
 | |
| 			// Check if the action supports file/folder targets
 | |
| 
 | |
| 			else if ( actionInfo.hasAttribute( AttrAnyFilesFolders) == true) {
 | |
| 
 | |
| 				// Pass one path at a time to the action
 | |
| 
 | |
| 				for ( size_t i = 0; i < argList.numberOfStrings(); i++) {
 | |
| 
 | |
| 					// Create a path list with a single path
 | |
| 
 | |
| 					StringList pathList;
 | |
| 					pathList.addString( argList[i]);
 | |
| 
 | |
| 					// Run the action
 | |
| 
 | |
| 					runAction( alfresco, pathList, actionInfo);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// Action does not use targets, just run the action
 | |
| 
 | |
| 			else if ( actionInfo.allowsNoParameters()) {
 | |
| 
 | |
| 				// Run the action
 | |
| 
 | |
| 				StringList emptyList;
 | |
| 				runAction( alfresco, emptyList, actionInfo);
 | |
| 			}
 | |
| 		}
 | |
| 		catch (Exception ex) {
 | |
| 			CString msg;
 | |
| 			msg.FormatMessage( L"Exception occurred\n\n%1", ex.getMessage().data());
 | |
| 			AfxMessageBox( msg, MB_OK | MB_ICONSTOP);
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		AfxMessageBox( L"Not a valid Alfresco CIFS folder", MB_OK | MB_ICONSTOP);
 | |
| 		DBGOUT_TS << "Error, not a valid Alfresco CIFS folder, " << folderPath << endl;
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	//	Exit the application
 | |
| 
 | |
| 	return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Process the command line arguments and build the parameter list for the desktop action
 | |
|  *
 | |
|  * @param alfresco AlfrescoInterface&
 | |
|  * @param paths StringList&
 | |
|  * @param actionInfo AlfrescoActionInfo&
 | |
|  * @param params DesktopParams&
 | |
|  * @return bool
 | |
|  */
 | |
| bool CAlfrescoApp::buildDesktopParameters( AlfrescoInterface& alfresco, StringList& paths, AlfrescoActionInfo& actionInfo,
 | |
| 										    DesktopParams& params) {
 | |
| 
 | |
| 	// If there are no paths then just return a success
 | |
| 
 | |
|     if ( paths.numberOfStrings() == 0)
 | |
| 	    return true;
 | |
| 
 | |
| 	// Process the list of files and either check in the file if it is a working copy or check out
 | |
| 	// the file
 | |
| 
 | |
| 	for ( unsigned int i = 0; i < paths.numberOfStrings(); i++) {
 | |
| 
 | |
| 		// Get the current file name
 | |
| 
 | |
| 		String curFile = paths.getStringAt( i);
 | |
| 
 | |
| 		// DEBUG
 | |
| 
 | |
| 		DBGOUT_TS << "Parameter: " << curFile << endl;
 | |
| 
 | |
| 		// Check if the path is on an Alfresco mapped drive
 | |
| 
 | |
| 		if ( alfresco.isMappedDrive() && curFile.startsWithIgnoreCase( alfresco.getDrivePath())) {
 | |
| 
 | |
| 			// Convert the path to a UNC path
 | |
| 
 | |
| 			String uncPath = alfresco.getRootPath();
 | |
| 			uncPath.append( curFile.substring(3));
 | |
| 
 | |
| 			curFile = uncPath;
 | |
| 		}
 | |
| 
 | |
| 		// Check if the path is to a file/folder, and whether it is a local path
 | |
| 
 | |
| 		bool copyFile = false;
 | |
| 		DWORD attr = GetFileAttributes( curFile);
 | |
| 
 | |
| 		if ( attr != INVALID_FILE_ATTRIBUTES) {
 | |
| 			
 | |
| 			// Check if the action supports the file/folder type
 | |
| 
 | |
| 			bool isDir = (attr & FILE_ATTRIBUTE_DIRECTORY) != 0 ? true : false;
 | |
| 
 | |
| 			if ( isDir && actionInfo.supportsFolders() == false) {
 | |
| 				AfxMessageBox(L"Action does not support folders", MB_OK | MB_ICONSTOP);
 | |
| 				DBGOUT_TS << "Error, action does not support folders" << endl;
 | |
| 				return false;
 | |
| 			}
 | |
| 			else if ( actionInfo.supportsFiles() == false) {
 | |
| 				AfxMessageBox(L"Action does not support files", MB_OK | MB_ICONSTOP);
 | |
| 				DBGOUT_TS << "Error, action does not support files" << endl;
 | |
| 				return false;
 | |
| 			}
 | |
| 
 | |
| 			// Get the file name from the path
 | |
| 
 | |
| 			StringList nameParts = FileName::splitPath( curFile);
 | |
| 			String curName = nameParts.getStringAt( 1);
 | |
| 
 | |
| 			// If the path is to a file that is not on the Alfresco share the file will need to be copied,
 | |
| 			// after checking the status of a matching file in the Alfresco folder
 | |
| 
 | |
| 			if ( curFile.length() >= 3 && curFile.startsWithIgnoreCase( alfresco.getDrivePath()) == false && 
 | |
| 				 curFile.substring(1,3).equals( L":\\")) {
 | |
| 
 | |
| 				// Check if the action supports local files
 | |
| 
 | |
| 				if ( isDir == false && actionInfo.hasAttribute(AttrClientFiles) == false) {
 | |
| 					AfxMessageBox(L"Action does not support local files", MB_OK | MB_ICONSTOP);
 | |
| 					DBGOUT_TS << "Error, action does not support local files" << endl;
 | |
| 					return false;
 | |
| 				}
 | |
| 				else if ( isDir == true && actionInfo.hasAttribute(AttrClientFolders) == false) {
 | |
| 					AfxMessageBox(L"Action does not support local folders", MB_OK | MB_ICONSTOP);
 | |
| 					DBGOUT_TS << "Error, action does not support local folders" << endl;
 | |
| 					return false;
 | |
| 				}
 | |
| 
 | |
| 				// Check if there is an existing file in the Alfresco with the same name, check if the file is locked
 | |
| 				
 | |
| 				PTR_AlfrescoFileInfo fInfo = alfresco.getFileInformation( curName);
 | |
| 				if ( fInfo.get() != NULL) {
 | |
| 
 | |
| 					// There is an existing file in the Alfresco folder with the same name, check if it is locked
 | |
| 
 | |
| 					if ( fInfo->getLockType() != LockNone) {
 | |
| 						AfxMessageBox( L"Cannot copy file to Alfresco folder, destination file is locked", MB_OK | MB_ICONEXCLAMATION);
 | |
| 						DBGOUT_TS << "Error, cannot copy to Alfresco folder, destination file is locked" << endl;
 | |
| 						return false;
 | |
| 					}
 | |
| 					else if ( actionInfo.hasPreProcessAction(PreLocalToWorkingCopy) == true && fInfo->isWorkingCopy() == false) {
 | |
| 						AfxMessageBox( L"Cannot copy to Alfresco folder, destination must overwrite a working copy", MB_OK | MB_ICONEXCLAMATION);
 | |
| 						DBGOUT_TS << "Error, cannot copy to Alfresco folder, destination must overwrite a working copy" << endl;
 | |
| 						return false;
 | |
| 					}
 | |
| 				}
 | |
| 				else if ( actionInfo.hasPreProcessAction(PreLocalToWorkingCopy) == true) {
 | |
| 
 | |
| 					// Target folder does not contain a matching working copy of the local file
 | |
| 
 | |
| 					CString msg;
 | |
| 					msg.FormatMessage( L"No matching working copy for %1", curName.data());
 | |
| 					AfxMessageBox( msg, MB_OK | MB_ICONEXCLAMATION);
 | |
| 					DBGOUT_TS << "Error, no matching working copy for " << curName << endl;
 | |
| 					return false;
 | |
| 				}
 | |
| 
 | |
| 				// Copy the files/folders using the Windows shell
 | |
| 
 | |
| 				bool copyAborted = false;
 | |
| 
 | |
| 				if ( copyFilesUsingShell( curFile, alfresco.getUNCPath(), copyAborted) == false) {
 | |
| 
 | |
| 					// Check if the copy failed or the user aborted the copy
 | |
| 
 | |
| 					if ( copyAborted == false) {
 | |
| 
 | |
| 						// File copy failed
 | |
| 
 | |
| 						CString msg;
 | |
| 						msg.FormatMessage( isDir ? L"Failed to copy folder %1" : L"Failed to copy file %1", curFile.data());
 | |
| 
 | |
| 						AfxMessageBox( msg, MB_OK | MB_ICONSTOP);
 | |
| 						DBGOUT_TS << "Error, copy failed for " << curName << endl;
 | |
| 						return false;
 | |
| 					}
 | |
| 					else {
 | |
| 
 | |
| 						// User aborted the file copy
 | |
| 
 | |
| 						CString msg;
 | |
| 						msg.FormatMessage( L"Copy aborted for %1", curFile.data());
 | |
| 						AfxMessageBox( msg, MB_OK | MB_ICONSTOP);
 | |
| 						DBGOUT_TS << "Error, copy aborted by user, " << curName << endl;
 | |
| 						return false;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "Added target " << curName << endl;
 | |
| 
 | |
| 				// Add a desktop target for the copied file
 | |
| 
 | |
| 				params.addTarget( new DesktopTarget(isDir ? TargetCopiedFolder : TargetCopiedFile, curName));
 | |
| 			}
 | |
| 			else {
 | |
| 
 | |
| 				//	Path is a UNC path, check if the file/folder is in the same folder as the action
 | |
| 
 | |
| 				DesktopTarget* pTarget = NULL;
 | |
| 
 | |
| 				if ( curFile.startsWith( alfresco.getUNCPath())) {
 | |
| 
 | |
| 					// Path is in the same folder as the application, or in a sub-folder
 | |
| 
 | |
| 					String relPath = curFile.substring( alfresco.getUNCPath().length() + 1);
 | |
| 
 | |
| 					if ( relPath.indexOf( L"\\") == -1) {
 | |
| 
 | |
| 						// Create a target using the file name only
 | |
| 
 | |
| 						pTarget = new DesktopTarget( isDir ? TargetFolder : TargetFile, relPath);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// If the target is not valid the file/folder is not in the same folder as the client-side application,
 | |
| 				// copy the files/folders to the target folder or use the root relative path to the file/folder
 | |
| 
 | |
| 				if ( pTarget == NULL) {
 | |
| 
 | |
| 					// Check if Alfresco files/folders should be copied to the target folder
 | |
| 
 | |
| 					if ( actionInfo.hasPreProcessAction(PreCopyToTarget)) {
 | |
| 
 | |
| 						// Copy the files/folders using the Windows shell
 | |
| 
 | |
| 						bool copyAborted = false;
 | |
| 
 | |
| 						if ( copyFilesUsingShell( curFile, alfresco.getUNCPath(), copyAborted) == false) {
 | |
| 
 | |
| 							// Check if the copy failed or the user aborted the copy
 | |
| 
 | |
| 							if ( copyAborted == false) {
 | |
| 
 | |
| 								// File copy failed
 | |
| 
 | |
| 								CString msg;
 | |
| 								msg.FormatMessage( isDir ? L"Failed to copy folder %1" : L"Failed to copy file %1", curFile.data());
 | |
| 
 | |
| 								AfxMessageBox( msg, MB_OK | MB_ICONSTOP);
 | |
| 								DBGOUT_TS << "Error, copy failed for " << curName << endl;
 | |
| 								return false;
 | |
| 							}
 | |
| 							else {
 | |
| 
 | |
| 								// User aborted the file copy
 | |
| 
 | |
| 								CString msg;
 | |
| 								msg.FormatMessage( L"Copy aborted for %1", curFile.data());
 | |
| 								AfxMessageBox( msg, MB_OK | MB_ICONSTOP);
 | |
| 								DBGOUT_TS << "Error, copy aborted for " << curName << endl;
 | |
| 								return false;
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						// Add a desktop target for the copied file
 | |
| 
 | |
| 						pTarget= new DesktopTarget(isDir ? TargetCopiedFolder : TargetCopiedFile, curName);
 | |
| 					}
 | |
| 					else {
 | |
| 
 | |
| 						// Get the root relative path to the file/folder
 | |
| 
 | |
| 						String rootRelPath = curFile.substring(alfresco.getRootPath().length());
 | |
| 						pTarget = new DesktopTarget( isDir ? TargetFolder : TargetFile, rootRelPath);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "Added target " << pTarget->getTarget() << endl;
 | |
| 
 | |
| 				// Add the desktop target
 | |
| 
 | |
| 				params.addTarget( pTarget);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Return status
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Copy a file/folder using the Windows shell
 | |
|  *
 | |
|  * @param fromFileFolder const String&
 | |
|  * @param toFolder const String&
 | |
|  * @param aborted bool&
 | |
|  * @return bool
 | |
|  */
 | |
| bool CAlfrescoApp::copyFilesUsingShell(const String& fromFileFolder, const String& toFolder, bool& aborted) {
 | |
| 
 | |
| 	// DEBUG
 | |
| 
 | |
| 	DBGOUT_TS << "Copy from " << fromFileFolder << " to " << toFolder << endl;
 | |
| 
 | |
| 	// Build the from/to paths, must be double null terminated
 | |
| 
 | |
| 	wchar_t fromPath[MAX_PATH + 1];
 | |
| 	wchar_t toPath[MAX_PATH + 1];
 | |
| 
 | |
| 	memset( fromPath, 0, sizeof( fromPath));
 | |
| 	memset( toPath, 0, sizeof( toPath));
 | |
| 
 | |
| 	wcscpy( fromPath, fromFileFolder.data());
 | |
| 	wcscpy( toPath, toFolder.data());
 | |
| 
 | |
| 	// Copy the local file to the Alfresco folder
 | |
| 
 | |
| 	SHFILEOPSTRUCT fileOpStruct;
 | |
| 	memset( &fileOpStruct, 0, sizeof(SHFILEOPSTRUCT));
 | |
| 
 | |
| 	fileOpStruct.hwnd  = HWND_DESKTOP;
 | |
| 	fileOpStruct.wFunc = FO_COPY;
 | |
| 	fileOpStruct.pFrom = fromPath;
 | |
| 	fileOpStruct.pTo   = toPath;
 | |
| 	fileOpStruct.fFlags= 0;
 | |
| 	fileOpStruct.fAnyOperationsAborted =false;
 | |
| 
 | |
| 	// Copy the file to the Alfresco folder
 | |
| 
 | |
| 	bool sts = false;
 | |
| 
 | |
| 	if ( SHFileOperation( &fileOpStruct) == 0) {
 | |
| 
 | |
| 		// File copy successful
 | |
| 
 | |
| 		sts = true;
 | |
| 	}
 | |
| 	else if ( fileOpStruct.fAnyOperationsAborted) {
 | |
| 
 | |
| 		// User aborted the file copy
 | |
| 
 | |
| 		aborted = true;
 | |
| 	}
 | |
| 
 | |
| 	// Return the copy status
 | |
| 
 | |
| 	return sts;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Run an action
 | |
|  *
 | |
|  * @param alfresco AlfrescoInterface&
 | |
|  * @param pathList StringList&
 | |
|  * @param actionInfo AlfrescoActionInfo&
 | |
|  * @return bool
 | |
|  */
 | |
| bool CAlfrescoApp::runAction( AlfrescoInterface& alfresco, StringList& pathList, AlfrescoActionInfo& actionInfo) {
 | |
| 
 | |
| 	// Build the desktop action parameter list, perform any file copying of local files
 | |
| 
 | |
| 	bool sts = false;
 | |
| 	DesktopParams desktopParams;
 | |
| 	
 | |
| 	if ( buildDesktopParameters( alfresco, pathList, actionInfo, desktopParams)) {
 | |
| 
 | |
| 		// Check if the action requires parameters
 | |
| 
 | |
| 		if ( actionInfo.allowsNoParameters() == false && desktopParams.numberOfTargets() == 0) {
 | |
| 			AfxMessageBox( L"No parameters for action", MB_OK | MB_ICONEXCLAMATION);
 | |
| 			DBGOUT_TS << "Error, no parameters for action" << endl;
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		// Run the desktop action
 | |
| 
 | |
| 		DesktopResponse response = alfresco.runAction( actionInfo, desktopParams);
 | |
| 
 | |
| 		// Check the response status
 | |
| 
 | |
| 		if ( response.getStatus() != StsSuccess) {
 | |
| 
 | |
| 			// Check if the status indicates a command line should be launched
 | |
| 
 | |
| 			if ( response.getStatus() == StsCommandLine) {
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "Action returned command line, " << response.getStatusMessage() << endl;
 | |
| 
 | |
| 				// Launch a process using the command line
 | |
| 
 | |
| 				sts = doCommandLine( alfresco, response.getStatusMessage());
 | |
| 			}
 | |
| 
 | |
| 			// Check if a web browser should be launched with a URL
 | |
| 
 | |
| 			else if ( response.getStatus() == StsLaunchURL) {
 | |
| 
 | |
| 				// DEBUG
 | |
| 
 | |
| 				DBGOUT_TS << "Action returned URL, " << response.getStatusMessage() << endl;
 | |
| 
 | |
| 				// Use the Windows shell to open the URL
 | |
| 
 | |
| 				sts = doURL( alfresco, response.getStatusMessage());
 | |
| 			}
 | |
| 
 | |
| 			// Error status
 | |
| 
 | |
| 			else {
 | |
| 
 | |
| 				//	Get the error message
 | |
| 
 | |
| 				String errMsg;
 | |
| 
 | |
| 				switch ( response.getStatus()) {
 | |
| 					case StsFileNotFound:
 | |
| 						errMsg = L"File not found";
 | |
| 						break;
 | |
| 					case StsAccessDenied:
 | |
| 						errMsg = L"Access denied";
 | |
| 						break;
 | |
| 					case StsBadParameter:
 | |
| 						errMsg = L"Bad parameter in request";
 | |
| 						break;
 | |
| 					case StsNoSuchAction:
 | |
| 						errMsg = L"No such action";
 | |
| 						break;
 | |
| 					default:
 | |
| 						errMsg = L"Error running action";
 | |
| 						break;
 | |
| 				}
 | |
| 
 | |
| 				// Display an error dialog
 | |
| 
 | |
| 				CString msg;
 | |
| 
 | |
| 				if ( response.hasStatusMessage())
 | |
| 					msg.FormatMessage( L"%1\n\n%2", errMsg.data(), response.getStatusMessage().data());
 | |
| 				else
 | |
| 					msg = errMsg.data();
 | |
| 
 | |
| 				AfxMessageBox( msg, MB_OK | MB_ICONERROR);
 | |
| 
 | |
| 				DBGOUT_TS << "Action returned error status, " << msg << endl;
 | |
| 			}
 | |
| 		}
 | |
| 		else if ( response.hasStatusMessage()) {
 | |
| 
 | |
| 			// Display the message returned by the action
 | |
| 
 | |
| 			CString msg;
 | |
| 			msg.FormatMessage( L"Action returned message\n\n%1", response.getStatusMessage().data());
 | |
| 			AfxMessageBox( msg, MB_OK | MB_ICONINFORMATION);
 | |
| 
 | |
| 			DBGOUT_TS << "Action returned error message, " << msg << endl;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Return the action status
 | |
| 
 | |
| 	return sts;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Launch a command line
 | |
|  *
 | |
|  * @param alfresco AlfrescoInterface&
 | |
|  * @param cmdStr const String&
 | |
|  * @return bool
 | |
|  */
 | |
| bool CAlfrescoApp::doCommandLine( AlfrescoInterface& alfresco, const String& cmdStr) {
 | |
| 
 | |
| 	// Check if the command line contains any environment variables/tokens
 | |
| 
 | |
| 	String cmdLine = cmdStr;
 | |
| 	int pos = cmdLine.indexOf( L'%');
 | |
| 
 | |
| 	if ( pos != -1) {
 | |
| 
 | |
| 		// Command line contains environment variables or other tokens that must be replaced
 | |
| 
 | |
| 		String newCmdLine = L"";
 | |
| 		if (pos > 0)
 | |
| 			newCmdLine = cmdLine.substring( 0, pos);
 | |
| 
 | |
| 		wchar_t envBuf[256];
 | |
| 		size_t envLen;
 | |
| 
 | |
| 		while ( pos != -1) {
 | |
| 
 | |
| 			// Find the end of the current token
 | |
| 
 | |
| 			int endPos = cmdLine.indexOf ( L'%', pos + 1);
 | |
| 
 | |
| 			if ( endPos == -1) {
 | |
| 				CString msg;
 | |
| 				msg.FormatMessage( L"Bad token in command line\n\n%1", cmdLine.data());
 | |
| 				AfxMessageBox( msg, MB_OK | MB_ICONERROR);
 | |
| 
 | |
| 				return false;
 | |
| 			}
 | |
| 
 | |
| 			// Extract the token
 | |
| 
 | |
| 			String token = cmdLine.substring( pos + 1, endPos);
 | |
| 
 | |
| 			// Replace the token with an environment variable value or other values
 | |
| 
 | |
| 			if ( token.equals( L"AlfrescoDir")) {
 | |
| 
 | |
| 				// Use the local path to the Alfresco folder that the application is running from
 | |
| 
 | |
| 				newCmdLine.append( alfresco.getUNCPath());
 | |
| 			}
 | |
| 			else {
 | |
| 
 | |
| 				// Find the environment variable value
 | |
| 
 | |
| 				envLen = sizeof( envBuf)/sizeof(wchar_t);
 | |
| 				const wchar_t* pEnvName = token.data();
 | |
| 
 | |
| 				if ( _wgetenv_s( &envLen, envBuf, envLen, pEnvName) == 0) {
 | |
| 
 | |
| 					// Append the environment variable value
 | |
| 
 | |
| 					newCmdLine.append( envBuf);
 | |
| 				}
 | |
| 				else {
 | |
| 
 | |
| 					// Error converting the environment variable
 | |
| 
 | |
| 					CString msg;
 | |
| 					msg.FormatMessage( L"Failed to convert environment variable\n\n%1\n\n%2", token.data(), cmdLine.data());
 | |
| 					AfxMessageBox( msg, MB_OK | MB_ICONERROR);
 | |
| 
 | |
| 					return false;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			// Update the token search position
 | |
| 
 | |
| 			pos = endPos + 1;
 | |
| 
 | |
| 			if (( unsigned int) pos < cmdStr.length()) {
 | |
| 
 | |
| 				// Search for the next token
 | |
| 
 | |
| 				pos = cmdLine.indexOf( L'%', pos);
 | |
| 			}
 | |
| 			else {
 | |
| 
 | |
| 				// End of string, finish the token search
 | |
| 
 | |
| 				pos = -1;
 | |
| 			}
 | |
| 
 | |
| 			// Append the normal string between tokens
 | |
| 
 | |
| 			if ( pos > (endPos + 1)) {
 | |
| 
 | |
| 				// Get the between token sting
 | |
| 
 | |
| 				String filler = cmdLine.substring( endPos + 1, pos);
 | |
| 				newCmdLine.append( filler);
 | |
| 			}
 | |
| 			else if ( pos == -1) {
 | |
| 
 | |
| 				// Append the remaining string
 | |
| 
 | |
| 				String filler = cmdLine.substring( endPos + 1);
 | |
| 				newCmdLine.append( filler);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Update the command line
 | |
| 
 | |
| 		cmdLine = newCmdLine;
 | |
| 	}
 | |
| 
 | |
| 	// Initialize the startup information
 | |
| 
 | |
| 	STARTUPINFO startupInfo;
 | |
| 	memset(&startupInfo, 0, sizeof(STARTUPINFO));
 | |
| 
 | |
| 	// Launch a process using the command line
 | |
| 
 | |
| 	PROCESS_INFORMATION processInfo;
 | |
| 	memset(&processInfo, 0, sizeof(PROCESS_INFORMATION));
 | |
| 
 | |
| 	bool sts = false;
 | |
| 
 | |
| 	if ( CreateProcess( NULL, (LPWSTR) cmdLine.data(), NULL, NULL, true, 0, NULL, NULL,
 | |
| 			&startupInfo, &processInfo) == false) {
 | |
| 		CString msg;
 | |
| 		msg.FormatMessage( L"Failed to launch command line\n\n%1\n\nError %2!d!", cmdLine.data(), GetLastError());
 | |
| 		AfxMessageBox( msg, MB_OK | MB_ICONERROR);
 | |
| 
 | |
| 		// DEBUG
 | |
| 
 | |
| 		DBGOUT_TS << "Error, failed to launch command line, status " << GetLastError() << endl;
 | |
| 	}
 | |
| 	else
 | |
| 		sts = true;
 | |
| 
 | |
| 	return sts;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Browse to a URL
 | |
|  *
 | |
|  * @param alfresco AlfrescoInterface&
 | |
|  * @param url const String&
 | |
|  * @return bool
 | |
|  */
 | |
| bool CAlfrescoApp::doURL( AlfrescoInterface& alfresco, const String& url) {
 | |
| 
 | |
| 	// Use the Windows shell to open the URL
 | |
| 
 | |
| 	bool sts = false;
 | |
| 
 | |
| 	HINSTANCE shellSts = ShellExecute( NULL, NULL, url.data(), NULL, NULL, SW_SHOWNORMAL);
 | |
| 	if (( int) shellSts < 32) {
 | |
| 		CString msg;
 | |
| 		msg.FormatMessage( L"Failed to launch URL\n\n%1", url.data());
 | |
| 		AfxMessageBox( msg, MB_OK | MB_ICONERROR);
 | |
| 	}
 | |
| 	else
 | |
| 		sts = true;
 | |
| 
 | |
| 	return sts;
 | |
| }
 |