Blob Blame Raw


#include "tsystem.h"
#include "tconvert.h"

#ifdef _WIN32
#define UNICODE // per le funzioni di conversione da/a UNC
#include <windows.h>
#include <lm.h>
#endif

//-----------------------------------------------------------------------------

bool TSystem::isUNC(const TFilePath &path)
{
	// si assume che il path e' gia' in formato UNC se inizia con "\\"
	std::wstring pathStr = path.getWideString();
	return pathStr.length() > 2 && pathStr.substr(0, 2) == L"\\\\";
}

//------------------------------------------------------------------------------

TFilePath TSystem::toUNC(const TFilePath &fp)
{
#ifdef _WIN32
	if (QString::fromStdWString(fp.getWideString()).startsWith('+'))
		return fp;
	if (isUNC(fp))
		return fp;

	std::string fpStr = ::to_string(fp);

	if (fpStr.length() > 1 && fpStr.c_str()[1] == ':') {
		std::string drive = fpStr.substr(0, 3);
		UINT uiDriveType = GetDriveTypeA(drive.c_str());

		if (uiDriveType & DRIVE_REMOTE) {
			// il drive e' montato

			DWORD cbBuff = _MAX_PATH; // Size of buffer
			char szBuff[_MAX_PATH];   // Buffer to receive information
			REMOTE_NAME_INFO *prni = (REMOTE_NAME_INFO *)&szBuff;

			// Pointers to head of buffer
			UNIVERSAL_NAME_INFO *puni = (UNIVERSAL_NAME_INFO *)&szBuff;

			DWORD dwResult = WNetGetUniversalNameW(::to_wstring(fpStr).c_str(),
												   UNIVERSAL_NAME_INFO_LEVEL,
												   (LPVOID)&szBuff,
												   &cbBuff);

			switch (dwResult) {
			case NO_ERROR:
				return TFilePath(::to_string(puni->lpUniversalName));

			case ERROR_NOT_CONNECTED:
				// The network connection does not exists.
				throw TException("The path specified doesn't refer to a shared folder");

			case ERROR_CONNECTION_UNAVAIL:
				// The device is not currently connected,
				// but it is a persistent connection.
				throw TException("The shared folder is not currently connected");

			default:
				throw TException("Cannot convert the path specified to UNC");
			}
		} else {
			//  It's a local drive so search for a share to it...
			NET_API_STATUS res;
			PSHARE_INFO_502 BufPtr, p;

			DWORD er = 0,
				  tr = 0,
				  resume = 0,
				  i;

			int iBestMatch = 0;

			std::string csTemp,
				csTempDrive,
				csBestMatch;

			do {
				res = NetShareEnum(NULL, 502, (LPBYTE *)&BufPtr, MAX_PREFERRED_LENGTH, &er, &tr, &resume);

				// If the call succeeds,
				if (res == ERROR_SUCCESS || res == ERROR_MORE_DATA) {
					p = BufPtr;

					// Loop through the entries;
					for (i = 1; i <= er; i++) {
						if (p->shi502_type == STYPE_DISKTREE) {
							//#ifdef IS_DOTNET
							// shi502_path e' una wstring, aanche se la dichiarazione di PSHARE_INFO_502 non lo sa!
							std::wstring shareLocalPathW = (LPWSTR)(p->shi502_path);
							std::string shareLocalPath = ::to_string(shareLocalPathW);
							//#else
							//string shareLocalPath = toString(p->shi502_path);
							//#endif

							if (toLower(fpStr).find(toLower(shareLocalPath)) == 0) {
								std::string hostName = TSystem::getHostName().toStdString();
								//   #ifdef IS_DOTNET
								// shi502_netname e' una wstring, anche se la dichiarazione di PSHARE_INFO_502 non lo sa!
								std::wstring shareNetNameW = (LPWSTR)(p->shi502_netname);
								std::string shareNetName = ::to_string(shareNetNameW);
								//	 #else
								//string shareNetName = toString(p->shi502_netname);
								//#endif
								shareNetName.append("\\");

								std::string fp(fpStr);
								std::string uncpath = "\\\\" + hostName + "\\" +
												 fp.replace(0, shareLocalPath.size(), shareNetName);

								return TFilePath(uncpath);
							}
						}

						p++;
					}

					// Free the allocated buffer.
					NetApiBufferFree(BufPtr);
				} else {
					//TRACE(_T("Error: %ld\n"), res);
				}

				// Continue to call NetShareEnum while
				//  there are more entries.
				//
			} while (res == ERROR_MORE_DATA); // end do
		}
	}

	//throw TException("Cannot convert the path specified to UNC");
	return fp;
#else
	//throw TException("Cannot convert the path specified to UNC");
	return fp;
#endif
}

//-----------------------------------------------------------------------------

TFilePath TSystem::toLocalPath(const TFilePath &fp)
{
#ifdef _WIN32
	if (!isUNC(fp))
		return TFilePath(fp);

	std::string pathStr = ::to_string(fp);

	// estrae hostname e il nome dello share dal path UNC
	std::string::size_type idx = pathStr.find_first_of("\\", 2);
	if (idx == std::string::npos)
		throw TException("The path specified is not a valid UNC path");

	std::string hostname = pathStr.substr(2, idx - 2);

	if (toLower(hostname) != toLower(TSystem::getHostName().toStdString()))
		throw TException("The UNC path specified does not refer to the local host");

	std::string::size_type idx2 = pathStr.find_first_of("\\", idx + 1);
	if (idx2 == std::string::npos)
		throw TException("The path specified is not a valid UNC path");

	std::string fpShareName = pathStr.substr(idx + 1, idx2 - idx - 1);
	std::string path = pathStr.substr(idx2 + 1, pathStr.size() - idx2);

	NET_API_STATUS res;
	do {
		PSHARE_INFO_502 BufPtr, p;

		DWORD er = 0, tr = 0, resume = 0;
		res = NetShareEnum(NULL, 502, (LPBYTE *)&BufPtr, DWORD(-1), &er, &tr, &resume);

		// If the call succeeds,
		if (res == ERROR_SUCCESS || res == ERROR_MORE_DATA) {
			p = BufPtr;

			// Loop through the entries;
			for (int i = 1; i <= (int)er; i++) {
				if (p->shi502_type == STYPE_DISKTREE) {
					//#ifdef IS_DOTNET
					//shi502_netname e' una wstring, anche se la dichiarazione di PSHARE_INFO_502 non lo sa!
					std::wstring shareNetNameW = (LPWSTR)(p->shi502_netname);
					std::string shareNetName = ::to_string(shareNetNameW);
					//	#else
					//string shareNetName = toString(p->shi502_netname);
					//#endif

					if (toLower(fpShareName) == toLower(shareNetName)) {
						//#ifdef IS_DOTNET
						// shi502_path e' una wstring, anche se la dichiarazione di PSHARE_INFO_502 non lo sa!
						std::wstring shareLocalPathW = (LPWSTR)(p->shi502_path);
						std::string shareLocalPath = ::to_string(shareLocalPathW);
						//#else
						//string shareLocalPath = toString(p->shi502_path);
						//#endif
						std::string localPath = shareLocalPath + path;
						return TFilePath(localPath);
					}
				}

				p++;
			}

			// Free the allocated buffer.
			NetApiBufferFree(BufPtr);
		} else {
			//TRACE(_T("Error: %ld\n"), res);
		}

		// Continue to call NetShareEnum while
		//  there are more entries.
		//
	} while (res == ERROR_MORE_DATA); // end do

	throw TException("Cannot convert to a local path");

#else
	throw TException("Cannot convert to a local path");
#endif
}