Initial commit.
This commit is contained in:
		
							
								
								
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| CMakeCache.txt | ||||
| CMakeFiles | ||||
| CMakeScripts | ||||
| Testing | ||||
| Makefile | ||||
| cmake_install.cmake | ||||
| install_manifest.txt | ||||
| compile_commands.json | ||||
| CTestTestfile.cmake | ||||
| build/ | ||||
| .vscode/* | ||||
							
								
								
									
										76
									
								
								AMTHIClient/Include/GetControlModeCommand.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								AMTHIClient/Include/GetControlModeCommand.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| /* SPDX-License-Identifier: Apache-2.0 */ | ||||
| /* | ||||
|  * Copyright (C) 2010-2019 Intel Corporation | ||||
|  */ | ||||
| /*++ | ||||
|  | ||||
| @file: GetControlModeCommand.h | ||||
|  | ||||
| --*/ | ||||
|  | ||||
| #ifndef __GET_CONTROL_MODE_COMMAND_H__ | ||||
| #define __GET_CONTROL_MODE_COMMAND_H__ | ||||
|  | ||||
| #include "AMTHICommand.h" | ||||
| #include "MEIparser.h" | ||||
|  | ||||
| namespace Intel | ||||
| { | ||||
| namespace MEI_Client | ||||
| { | ||||
| namespace AMTHI_Client | ||||
| { | ||||
| 	struct GET_CONTROL_MODE_RESPONSE | ||||
| 	{ | ||||
| 		uint32_t   ControlMode; | ||||
|  | ||||
| 		void parse (std::vector<uint8_t>::const_iterator& itr, const std::vector<uint8_t>::const_iterator end) | ||||
| 		{ | ||||
| 			Intel::MEI_Client::parseData(*this, itr, end); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	class GetControlModeRequest; | ||||
| 	class GetControlModeCommand : public AMTHICommand | ||||
| 	{ | ||||
| 	public: | ||||
|  | ||||
| 		GetControlModeCommand(); | ||||
| 		virtual ~GetControlModeCommand() {} | ||||
|  | ||||
| 		virtual void reTransact(); | ||||
| 		GET_CONTROL_MODE_RESPONSE getResponse(); | ||||
|  | ||||
| 	private: | ||||
| 		virtual void parseResponse(const std::vector<uint8_t>& buffer); | ||||
|  | ||||
| 		std::shared_ptr<AMTHICommandResponse<GET_CONTROL_MODE_RESPONSE>> m_response; | ||||
|  | ||||
| 		static const uint32_t RESPONSE_COMMAND_NUMBER = 0x0480006B; | ||||
| 	}; | ||||
|  | ||||
| 	class GetControlModeRequest : public AMTHICommandRequest | ||||
| 	{ | ||||
| 	public: | ||||
| 		GetControlModeRequest() {} | ||||
| 		virtual ~GetControlModeRequest() {} | ||||
|  | ||||
| 	private: | ||||
| 		static const uint32_t REQUEST_COMMAND_NUMBER = 0x0400006B; | ||||
| 		virtual unsigned int requestHeaderCommandNumber() | ||||
| 		{ | ||||
| 			//this is the command number (taken from the AMTHI document) | ||||
| 			return REQUEST_COMMAND_NUMBER; | ||||
| 		} | ||||
|  | ||||
| 		virtual uint32_t requestDataSize() | ||||
| 		{ | ||||
| 			return 0; | ||||
| 		} | ||||
| 		virtual std::vector<uint8_t> SerializeData(); | ||||
| 	}; | ||||
| } // namespace AMTHI_Client | ||||
| } // namespace MEI_Client | ||||
| } // namespace Intel | ||||
|  | ||||
| #endif //__GET_CONTROL_MODE_COMMAND_H__ | ||||
							
								
								
									
										51
									
								
								AMTHIClient/Src/GetControlModeCommand.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								AMTHIClient/Src/GetControlModeCommand.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | ||||
| /* SPDX-License-Identifier: Apache-2.0 */ | ||||
| /* | ||||
|  * Copyright (C) 2010-2019 Intel Corporation | ||||
|  */ | ||||
| /*++ | ||||
|  | ||||
| @file: GetControlModeCommand.cpp | ||||
|  | ||||
| --*/ | ||||
|  | ||||
| #include "GetControlModeCommand.h" | ||||
| #include "StatusCodeDefinitions.h" | ||||
| #include <string.h> | ||||
|  | ||||
| using namespace std; | ||||
|  | ||||
| using namespace Intel::MEI_Client::AMTHI_Client; | ||||
|  | ||||
| GetControlModeCommand::GetControlModeCommand() | ||||
| { | ||||
| 	shared_ptr<MEICommandRequest> tmp(new GetControlModeRequest()); | ||||
| 	m_request = tmp; | ||||
| 	Transact(); | ||||
| } | ||||
|  | ||||
| void GetControlModeCommand::reTransact() | ||||
| { | ||||
| 	shared_ptr<MEICommandRequest> tmp(new GetControlModeRequest()); | ||||
| 	m_request = tmp; | ||||
| 	Transact(); | ||||
| } | ||||
|  | ||||
| GET_CONTROL_MODE_RESPONSE GetControlModeCommand::getResponse() | ||||
| { | ||||
| 	return m_response->getResponse(); | ||||
| } | ||||
|  | ||||
| void | ||||
| GetControlModeCommand::parseResponse(const vector<uint8_t>& buffer) | ||||
| { | ||||
| 	shared_ptr<AMTHICommandResponse<GET_CONTROL_MODE_RESPONSE>> tmp( | ||||
| 		new AMTHICommandResponse<GET_CONTROL_MODE_RESPONSE>(buffer, RESPONSE_COMMAND_NUMBER)); | ||||
| 	m_response = tmp; | ||||
| } | ||||
|  | ||||
| std::vector<uint8_t>  | ||||
| GetControlModeRequest::SerializeData() | ||||
| { | ||||
| 	vector<uint8_t> output; | ||||
| 	return output; | ||||
| } | ||||
							
								
								
									
										131
									
								
								CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | ||||
| cmake_minimum_required (VERSION 3.1) | ||||
|  | ||||
| project (rpc VERSION 1.0.0) | ||||
|  | ||||
| set (CMAKE_CXX_STANDARD 11) | ||||
|  | ||||
| # RPC version info | ||||
| configure_file(version.h.in | ||||
|                version.h) | ||||
| include_directories(${PROJECT_BINARY_DIR}) | ||||
|  | ||||
| # TODO: figure out how to read the LMS version from repo like the main lms CMakeLists.txt | ||||
| set (LMS_VERSION_STRING 1932.0.0.0) | ||||
|  | ||||
| # Compiler settings [Obtained from CmakeLists.txt for lms] | ||||
| string(APPEND CMAKE_CXX_FLAGS_DEBUG " -DDEBUG -D_DEBUG") | ||||
| string(APPEND CMAKE_C_FLAGS_DEBUG " -DDEBUG -D_DEBUG") | ||||
|  | ||||
| set (CMAKE_POSITION_INDEPENDENT_CODE ON) | ||||
|  | ||||
| if (UNIX) | ||||
|   set (CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -z noexecstack -z relro -z now") | ||||
|   set (CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -z noexecstack -z relro -z now") | ||||
|  | ||||
|   #CMake issue #14983 | ||||
|   set (CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -pie") | ||||
|  | ||||
|   #Secure library usage and secure compile flags | ||||
|   add_definitions (-fstack-protector-strong -D_FORTIFY_SOURCE=2 -O2 -Wformat -Wformat-security) | ||||
|   add_definitions (-fno-strict-overflow -fno-delete-null-pointer-checks -fwrapv) | ||||
| else (UNIX) | ||||
|   add_definitions (/GS /sdl) | ||||
|   set (CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} /NXCompat /DynamicBase") | ||||
|   #add_definitions (/D UNICODE /D _UNICODE) | ||||
|   add_definitions (/D UNICODE /D _UNICODE  /D_NO_ASYNCRTIMP /D_ASYNCRT_EXPORT /D_NO_PPLXIMP /DWIN32 /DMBCS /D_USRDLL /DCPPREST_EXCLUDE_COMPRESSION /D_WINSOCK_DEPRECATED_NO_WARNINGS) | ||||
|   add_compile_options ($<$<CONFIG:Release>:/O2>) | ||||
|   add_compile_options (/MT$<$<CONFIG:Debug>:d>) | ||||
|   set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /MT") | ||||
|   set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd") | ||||
| endif (UNIX) | ||||
|  | ||||
|  | ||||
| # Download and unpack lms at configure time | ||||
| configure_file(lms.cmake.in | ||||
|                lms-download/CMakeLists.txt) | ||||
| execute_process(COMMAND ${CMAKE_COMMAND} -G "${CMAKE_GENERATOR}" . | ||||
|   WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/lms-download ) | ||||
| execute_process(COMMAND ${CMAKE_COMMAND} --build . | ||||
|   WORKING_DIRECTORY ${CMAKE_BINARY_DIR}/lms-download ) | ||||
|  | ||||
| # Add  MEIClient directly to our build. This adds | ||||
| # the following targets: LmsMEIClient and LIBMETEE | ||||
| add_subdirectory(${CMAKE_BINARY_DIR}/lms-src/MEIClient | ||||
|                  ${CMAKE_BINARY_DIR}/lms-build/MEIClient) | ||||
| add_dependencies(LmsMEIClient libmetee) | ||||
|  | ||||
|  | ||||
| if (UNIX) | ||||
|  | ||||
| # Find threads [unix it pthreads] | ||||
| set(CMAKE_THREAD_PREFER_PTHREAD TRUE) | ||||
| set(THREADS_PREFER_PTHREAD_FLAG TRUE) | ||||
| find_package(Threads REQUIRED) | ||||
|  | ||||
| # Find Boost | ||||
| find_package(Boost COMPONENTS system REQUIRED) | ||||
|  | ||||
| # Find OpenSSL | ||||
| find_package(OpenSSL) | ||||
|  | ||||
| # Download and build CppRestSDK, If GIT_TAG is changed then need to delete cpprestsdk-prefix because UPDATE_COMMAND is set to "" | ||||
| include(ExternalProject) | ||||
| ExternalProject_Add(cpprestsdk | ||||
|   GIT_REPOSITORY    https://github.com/Microsoft/cpprestsdk.git  | ||||
|   GIT_TAG           v2.10.14 | ||||
|   CMAKE_ARGS        -DCMAKE_BUILD_TYPE=Release -DBUILD_SHARED_LIBS=0 -DBUILD_SAMPLES=OFF -DBUILD_TESTS=OFF -DCMAKE_INSTALL_PREFIX=<SOURCE_DIR>/../../install | ||||
|   TEST_COMMAND      "" | ||||
|   UPDATE_COMMAND    "" | ||||
| ) | ||||
| ExternalProject_Get_Property(cpprestsdk SOURCE_DIR) | ||||
| set(CPPRESTSDK_LIBARIES ${SOURCE_DIR}/../../install/lib/) | ||||
| set(CPPRESTSDK_INCLUDE_DIR ${SOURCE_DIR}/../../install/include/) | ||||
|  | ||||
| add_library(cpprest INTERFACE) | ||||
| target_link_libraries(cpprest INTERFACE ${CPPRESTSDK_LIBARIES}/libcpprest.a  OpenSSL::SSL OpenSSL::Crypto ${Boost_LIBRARIES} Threads::Threads) | ||||
| target_include_directories(cpprest INTERFACE ${CPPRESTSDK_INCLUDE_DIR}) | ||||
|  | ||||
| else (UNIX) | ||||
|  | ||||
| # CppRestSDK | ||||
| find_package(cpprestsdk CONFIG REQUIRED) | ||||
|  | ||||
| endif (UNIX) | ||||
|  | ||||
| # ccu-poc | ||||
| add_executable (rpc | ||||
|   AMTHIClient/Include/GetControlModeCommand.h | ||||
|   AMTHIClient/Src/GetControlModeCommand.cpp | ||||
|   commands.h | ||||
|   commands.cpp | ||||
|   lms.h | ||||
|   lms.cpp | ||||
|   main.cpp | ||||
| ) | ||||
|  | ||||
| target_include_directories(rpc PUBLIC | ||||
|   "AMTHIClient/Include/" | ||||
| ) | ||||
|  | ||||
| if (UNIX) | ||||
|  | ||||
| add_dependencies(rpc cpprestsdk) | ||||
|  | ||||
| target_link_libraries (rpc PRIVATE  | ||||
|   LmsMEIClient | ||||
|   cpprest | ||||
| ) | ||||
|  | ||||
| else (UNIX) | ||||
|  | ||||
| target_link_libraries (rpc PRIVATE  | ||||
|   LmsMEIClient | ||||
|   iphlpapi | ||||
|   cpprestsdk::cpprest | ||||
|   cpprestsdk::cpprestsdk_zlib_internal | ||||
|   cpprestsdk::cpprestsdk_boost_internal | ||||
|   cpprestsdk::cpprestsdk_brotli_internal | ||||
|   ${Boost_LIBRARIES} | ||||
| ) | ||||
|  | ||||
| endif (UNIX) | ||||
							
								
								
									
										67
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								README.md
									
									
									
									
									
								
							| @@ -1 +1,66 @@ | ||||
| # rpc | ||||
| # Remote Provisioning Client (RPC) | ||||
|  | ||||
| RPC is an application which enables remote capabilities for AMT, such as as device activation. To accomplish this, RPC communicates with the RPS (Remote Provisioning Server). | ||||
|  | ||||
| As a prerequisite, a Local Management Service (LMS) must be installed and running on the operating system. | ||||
|  | ||||
| ## Linux | ||||
|  | ||||
| Steps below are for Ubuntu 18.04. | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| - sudo apt install git cmake build-essential libboost-system-dev  libboost-thread-dev libboost-random-dev libboost-regex-dev libboost-filesystem-dev libssl-dev zlib1g-dev | ||||
|  | ||||
| ### Build | ||||
|  | ||||
| - mkdir build | ||||
| - cd build | ||||
| - cmake -DCMAKE_BUILD_TYPE=Release .. | ||||
|   - Build debug: cmake -DCMAKE_BUILD_TYPE=Debug .. | ||||
| - cmake --build . | ||||
|  | ||||
| ### Run | ||||
|  | ||||
| - See ./rpc --help for details. | ||||
| - Example | ||||
|   - sudo ./rpc --url wss://localhost:8080 --profile profile1 | ||||
|  | ||||
| ## Windows | ||||
|  | ||||
| Steps below are for Windows 10 and Visual Studio 2019 Professional. | ||||
|  | ||||
| ### Build VCPKG | ||||
|  | ||||
| Open an x64 native command prompt for Visual Studio 2019 as Administrator. | ||||
|  | ||||
| - git clone --branch 2020.01 https://github.com/microsoft/vcpkg.git | ||||
| - cd vcpkg | ||||
| - bootstrap-vcpkg.bat | ||||
| - vcpkg integrate install | ||||
|  | ||||
| ### Build C++ REST SDK | ||||
|  | ||||
| Open an x64 native tools command prompt for Visual Studio 2019. | ||||
|  | ||||
| - cd vcpkg | ||||
| - vcpkg install cpprestsdk:x64-windows-static | ||||
|  | ||||
| ### Build | ||||
|  | ||||
| Open an x64 native tools command prompt for Visual Studio 2019. | ||||
|  | ||||
| - mkdir build | ||||
| - cd build | ||||
| - cmake .. -DVCPKG_TARGET_TRIPLET=x64-windows-static -DCMAKE_TOOLCHAIN_FILE=/vcpkg/scripts/buildsystems/vcpkg.cmake | ||||
| - cmake --build . --config Release | ||||
|   - Build debug: cmake --build . --config Debug | ||||
|  | ||||
| ### Run | ||||
|  | ||||
| Open a command prompt as Administrator. | ||||
|  | ||||
| - See rpc.exe --help for details. | ||||
| - Example | ||||
|   - cd build\Release | ||||
|   - rpc.exe --url wss://localhost:8080 --profile profile1 | ||||
|   | ||||
							
								
								
									
										499
									
								
								commands.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										499
									
								
								commands.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,499 @@ | ||||
| /* | ||||
| Copyright 2019 Intel Corporation | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| 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 "commands.h" | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| #include <boost/asio.hpp> | ||||
| #include <winsock2.h> | ||||
| #include <iphlpapi.h> | ||||
| #endif | ||||
|  | ||||
| #include "AMTHICommand.h" | ||||
| #include "MEIClientException.h" | ||||
| #include "GetUUIDCommand.h" | ||||
| #include "GetLocalSystemAccountCommand.h" | ||||
| #include "GetCodeVersionCommand.h" | ||||
| #include "GetControlModeCommand.h" | ||||
| #include "GetProvisioningStateCommand.h" | ||||
| #include "GetDNSSuffixCommand.h" | ||||
| #include "GetLanInterfaceSettingsCommand.h" | ||||
| #include "GetCertificateHashEntryCommand.h" | ||||
| #include "EnumerateHashHandlesCommand.h" | ||||
| #include "MEIparser.h" | ||||
| #include "version.h" | ||||
| #include <boost/algorithm/string.hpp> | ||||
|  | ||||
| #include <cpprest/ws_client.h> | ||||
| #include <cpprest/json.h> | ||||
| #include <cpprest/streams.h> | ||||
| #include <sstream> | ||||
| #include <iostream> | ||||
| #include <string> | ||||
| #include "lms.h" | ||||
|  | ||||
| using namespace std; | ||||
| using namespace Intel::MEI_Client::AMTHI_Client; | ||||
| using namespace web::websockets::client; | ||||
| using namespace web; | ||||
|  | ||||
| #define WORKING_BUFFER_SIZE 15000 | ||||
| #define MAX_TRIES 3 | ||||
|  | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| std::string getDNSFromMAC(char *macAddress) | ||||
| { | ||||
|     std::string dnsSuffix = ""; | ||||
|     char dns[256]; | ||||
|     memset(dns, 0, 256); | ||||
|  | ||||
|     PIP_ADAPTER_ADDRESSES pAddresses = NULL; | ||||
|     DWORD dwSize = 0; | ||||
|     DWORD dwRetVal = 0; | ||||
|     ULONG outBufLen = 0; | ||||
|     ULONG Iterations = 0; | ||||
|     outBufLen = WORKING_BUFFER_SIZE; | ||||
|  | ||||
|     // get info for all adapters | ||||
|     do { | ||||
|  | ||||
|         pAddresses = (IP_ADAPTER_ADDRESSES *) malloc(outBufLen); | ||||
|         if (pAddresses == NULL) { | ||||
|             cout << "dns memory error" << std::endl; | ||||
|             return dnsSuffix; | ||||
|         } | ||||
|  | ||||
|         dwRetVal = GetAdaptersAddresses(AF_INET, GAA_FLAG_INCLUDE_PREFIX, NULL, pAddresses, &outBufLen); | ||||
|  | ||||
|         if (dwRetVal == ERROR_BUFFER_OVERFLOW)  | ||||
|         { | ||||
|             free(pAddresses); | ||||
|             pAddresses = NULL; | ||||
|         }  | ||||
|         else  | ||||
|         { | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         Iterations++; | ||||
|  | ||||
|     } while ((dwRetVal == ERROR_BUFFER_OVERFLOW) && (Iterations < MAX_TRIES)); | ||||
|  | ||||
|     // get DNS from MAC | ||||
|     PIP_ADAPTER_ADDRESSES pCurrAddresses = NULL; | ||||
|     pCurrAddresses = pAddresses; | ||||
|     while (pCurrAddresses)  | ||||
|     { | ||||
|         if (pCurrAddresses->PhysicalAddressLength != 0)  | ||||
|         { | ||||
|             if (memcmp(macAddress, (char *) pCurrAddresses->PhysicalAddress, 6) == 0) | ||||
|             { | ||||
|                 if (wcslen(pCurrAddresses->DnsSuffix) > 0) | ||||
|                 { | ||||
|                     snprintf(dns, 256, "%ws", pCurrAddresses->DnsSuffix ); | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         pCurrAddresses = pCurrAddresses->Next; | ||||
|     } | ||||
|  | ||||
|     dnsSuffix = dns; | ||||
|      | ||||
|     return dnsSuffix; | ||||
| } | ||||
|  | ||||
| #else | ||||
| std::string getDNSFromMAC(char *macAddress) | ||||
| { | ||||
|     std::string dnsSuffix = ""; | ||||
|  | ||||
|     // get socket | ||||
|     SOCKET s = 0; | ||||
|     s = socket(PF_INET, SOCK_DGRAM, 0); | ||||
|     if (s < 0)  | ||||
|     { | ||||
|         cout << "couldn't get socket" << endl; | ||||
|         return dnsSuffix; | ||||
|     } | ||||
|  | ||||
|     // get info for all adapters | ||||
|     struct ifconf ifc; | ||||
|     memset(&ifc, 0, sizeof(ifconf)); | ||||
|     char buffer[8192]; | ||||
|     memset(buffer, 0, sizeof(buffer)); | ||||
|  | ||||
|     ifc.ifc_buf = buffer; | ||||
|     ifc.ifc_len = sizeof(buffer); | ||||
|     if(ioctl(s, SIOCGIFCONF, &ifc) < 0)  | ||||
|     { | ||||
|         cout << "ioctl SIOCGIFCONF failed" << endl; | ||||
|         return dnsSuffix; | ||||
|     } | ||||
|  | ||||
|     // get DNS from IP associated with MAC | ||||
|     struct ifreq *ifr = ifc.ifc_req; | ||||
|     int interfaceCount = ifc.ifc_len / sizeof(struct ifreq); | ||||
|  | ||||
|     char ip[INET6_ADDRSTRLEN] = {0}; | ||||
|     struct ifreq *item; | ||||
|     struct sockaddr *addr; | ||||
|     for(int i = 0; i < interfaceCount; i++)  | ||||
|     { | ||||
|         item = &ifr[i]; | ||||
|         addr = &(item->ifr_addr); | ||||
|  | ||||
|         // get IP address | ||||
|         if(ioctl(s, SIOCGIFADDR, item) < 0)  | ||||
|         { | ||||
|             cout << "ioctl SIOCGIFADDR failed" << endl; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (inet_ntop(AF_INET, &( ((struct sockaddr_in *)addr)->sin_addr ),  | ||||
|             ip, sizeof(ip) ) == NULL) | ||||
|         { | ||||
|             cout << "inet_ntop" << endl; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         // get MAC address | ||||
|         if(ioctl(s, SIOCGIFHWADDR, item) < 0)  | ||||
|         { | ||||
|             cout << "ioctl SIOCGIFHWADDR failed" << endl; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (memcmp(macAddress, (char *) item->ifr_hwaddr.sa_data, 6) == 0) | ||||
|         { | ||||
|             // Get host by using the IP address which AMT device is using | ||||
|             struct in_addr addr = {0}; | ||||
|             struct hostent *host; | ||||
|             addr.s_addr = inet_addr(ip); | ||||
|             host = gethostbyaddr((char *)&addr, 4, AF_INET); | ||||
|             if (host == NULL) | ||||
|             { | ||||
|                 cout << "gethostbyaddr() failed"; | ||||
|                 return dnsSuffix; | ||||
|             } | ||||
|  | ||||
|             // strip off the hostname to get actual domain name | ||||
|             int domainNameSize = 256; | ||||
|             char domainName[domainNameSize]; | ||||
|             memset(domainName, 0, domainNameSize); | ||||
|              | ||||
|             char *dn = strchr(host->h_name, '.'); | ||||
|             if (dn != NULL) | ||||
|             { | ||||
|                 if (domainNameSize >= strlen(dn + 1)) | ||||
|                 { | ||||
|                     snprintf(domainName, domainNameSize, "%s", ++dn); | ||||
|  | ||||
|                     dnsSuffix = domainName; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return dnsSuffix; | ||||
| } | ||||
| #endif | ||||
|  | ||||
|  | ||||
| json::value getCertificateHashes() | ||||
| { | ||||
|     json::value certHashes; | ||||
|     vector<json::value> hashValues; | ||||
|  | ||||
|     // get the hash handles | ||||
|     EnumerateHashHandlesCommand command; | ||||
|     ENUMERATE_HASH_HANDLES_RESPONSE response = command.getResponse(); | ||||
|  | ||||
|     vector<unsigned int>::iterator itr = response.HashHandles.begin(); | ||||
|     vector<unsigned int>::iterator endItr = response.HashHandles.end(); | ||||
|     for (; itr != endItr; ++itr) | ||||
|     { | ||||
|         // get each entry | ||||
|         GetCertificateHashEntryCommand command(*itr); | ||||
|         GET_CERTIFICATE_HASH_ENTRY_RESPONSE response = command.getResponse(); | ||||
|  | ||||
|         int hashSize; | ||||
|         switch (response.HashAlgorithm) { | ||||
|             case 0: // MD5 | ||||
|                 hashSize = 16; | ||||
|                 break; | ||||
|             case 1: // SHA1 | ||||
|                 hashSize = 20; | ||||
|                 break; | ||||
|             case 2: // SHA256 | ||||
|                 hashSize = 32; | ||||
|                 break; | ||||
|             case 3: // SHA512 | ||||
|                 hashSize = 64; | ||||
|                 break; | ||||
|             default: | ||||
|                 hashSize = 64; | ||||
|                 break; | ||||
|         } | ||||
|          | ||||
|         if (response.IsActive == 1) {  | ||||
|             string hashString; | ||||
|             hashString.clear(); | ||||
|             for (int i = 0; i < hashSize; i++) | ||||
|             { | ||||
|                 char hex[10]; | ||||
|                 snprintf(hex, 10, "%02x", response.CertificateHash[i]); | ||||
|                 hashString += hex; | ||||
|             } | ||||
|  | ||||
|             hashValues.push_back( json::value::string(  utility::conversions::convertstring(hashString) ) ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return json::value::array(hashValues); | ||||
| } | ||||
|  | ||||
| std::string getDNSInfo() | ||||
| { | ||||
|     std::string dnsSuffix; | ||||
|  | ||||
|     // Get interface info which AMT is using. We don't worry about wireless since | ||||
|     // only wired used for configuration | ||||
|     GetLanInterfaceSettingsCommand getLanInterfaceSettingsCommandWired(Intel::MEI_Client::AMTHI_Client::WIRED); | ||||
|     LAN_SETTINGS lanSettings = getLanInterfaceSettingsCommandWired.getResponse(); | ||||
|  | ||||
|     if (!lanSettings.Enabled) | ||||
|     { | ||||
|         cout << "error: no wired AMT interfaces enabled" << endl; | ||||
|         return ""; | ||||
|     } | ||||
|  | ||||
|     // Get DNS according to AMT | ||||
|     GetDNSSuffixCommand getDnsSuffixCommand; | ||||
| 	dnsSuffix = getDnsSuffixCommand.getResponse(); | ||||
|  | ||||
|     // get DNS from OS | ||||
|     if (!dnsSuffix.length()) | ||||
|     { | ||||
|         dnsSuffix = getDNSFromMAC((char *)&lanSettings.MacAddress); | ||||
|     } | ||||
|  | ||||
|     return dnsSuffix; | ||||
| } | ||||
|  | ||||
|  | ||||
| string getActivateInfo(string profile) | ||||
| { | ||||
|     utility::string_t tmp; | ||||
|  | ||||
|     // Activation parameters | ||||
|     json::value activationParams; | ||||
|  | ||||
|     // Get code version | ||||
|     GetCodeVersionCommand codeVersionCommand; | ||||
|     CODE_VERSIONS codeVersion = codeVersionCommand.getResponse(); | ||||
|  | ||||
|     // Additional versions | ||||
|     // UINT8[16] UUID; | ||||
|     // AMT_VERSION_TYPE Version and Description are std::string. | ||||
|     for (vector<AMT_VERSION_TYPE>::iterator it = codeVersion.Versions.begin(); it != codeVersion.Versions.end(); it++) | ||||
|     { | ||||
|         if (boost::iequals(it->Description, "AMT"))  | ||||
|         { | ||||
|             tmp = utility::conversions::convertstring(it->Version); | ||||
|             activationParams[U("ver")] = json::value::string(tmp); | ||||
|         } | ||||
|         else if (boost::iequals(it->Description, "Build Number"))  | ||||
|         { | ||||
|             tmp = utility::conversions::convertstring(it->Version); | ||||
|             activationParams[U("build")] = json::value::string(tmp); | ||||
|         } | ||||
|         else if (boost::iequals(it->Description, "Sku"))  | ||||
|         { | ||||
|             tmp = utility::conversions::convertstring(it->Version); | ||||
|             activationParams[U("sku")] = json::value::string(tmp); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Get UUID | ||||
|     GetUUIDCommand get; | ||||
|     GET_UUID_RESPONSE res = get.getResponse(); | ||||
|     std::vector<json::value> UUID; | ||||
|     for (int i = 0; i < 16; i++) | ||||
|     { | ||||
|         UUID.push_back(json::value(res.UUID[i])); | ||||
|     } | ||||
|     activationParams[U("uuid")] = json::value::array(UUID); | ||||
|  | ||||
|     // Get local system account | ||||
|     // User name in ASCII char-set. The string is NULL terminated. CFG_MAX_ACL_USER_LENGTH is 33 | ||||
|     // Password in ASCII char set. From AMT 6.1 this field is in BASE64 format. The string is NULL terminated.  | ||||
|     GetLocalSystemAccountCommand sac; | ||||
|     tmp = utility::conversions::convertstring(sac.getResponse().UserName); | ||||
|     activationParams[U("username")] = json::value::string(tmp); | ||||
|     tmp = utility::conversions::convertstring(sac.getResponse().Password); | ||||
|     activationParams[U("password")] = json::value::string(tmp); | ||||
|  | ||||
|     // Get Control Mode | ||||
|     GetControlModeCommand controlModeCommand; | ||||
|     GET_CONTROL_MODE_RESPONSE controlMode = controlModeCommand.getResponse(); | ||||
|     activationParams[U("currentMode")] = json::value::number(controlMode.ControlMode); | ||||
|  | ||||
|     // Get DNS Info | ||||
|     tmp = utility::conversions::convertstring(""); | ||||
|     string dnsSuffix = getDNSInfo(); | ||||
|  | ||||
|     if (dnsSuffix.length()) | ||||
|     { | ||||
|         tmp = utility::conversions::convertstring(dnsSuffix); | ||||
|     } | ||||
|      | ||||
|     activationParams[U("fqdn")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("PPC"); | ||||
|     activationParams[U("client")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring(profile); | ||||
|     activationParams[U("profile")] = json::value::string(tmp); | ||||
|  | ||||
|     // Get certificate hashes | ||||
|     activationParams[U("certHashes")] = getCertificateHashes(); | ||||
|      | ||||
|     // Return serialized parameters in base64 | ||||
|     string serializedParams = utility::conversions::to_utf8string(activationParams.serialize()); | ||||
| #ifdef DEBUG | ||||
|     cout << "Activation info payload:" << serializedParams << std::endl; | ||||
| #endif | ||||
|  | ||||
|     return encodeBase64(serializedParams); | ||||
| } | ||||
|  | ||||
| string encodeBase64(string str) | ||||
| { | ||||
|     std::vector<unsigned char> strVector(str.begin(), str.end()); | ||||
|     utility::string_t base64 = utility::conversions::to_base64(strVector); | ||||
|     string encodedString = utility::conversions::to_utf8string(base64); | ||||
|  | ||||
|     return encodedString; | ||||
| } | ||||
|  | ||||
| string decodeBase64(string str) | ||||
| { | ||||
|     utility::string_t serializedData = utility::conversions::to_string_t(str); | ||||
|     std::vector<unsigned char> strVector = utility::conversions::from_base64(serializedData); | ||||
|     string decodedString(strVector.begin(), strVector.end()); | ||||
|      | ||||
|     return decodedString; | ||||
| } | ||||
|  | ||||
| string createActivationRequest(string profile) | ||||
| { | ||||
|     // Activation parameters | ||||
|     json::value request; | ||||
|  | ||||
|     // placeholder stuff; will likely change | ||||
|     utility::string_t tmp = utility::conversions::convertstring("activation"); | ||||
|     request[U("method")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("key"); | ||||
|     request[U("apiKey")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring(PROJECT_VER); | ||||
|     request[U("appVersion")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring(PROTOCOL_VERSION); | ||||
|     request[U("protocolVersion")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("ok"); | ||||
|     request[U("status")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("all\'s good!"); | ||||
|     request[U("message")] = json::value::string(tmp); | ||||
|  | ||||
|     // payload  | ||||
|     string activationInfo = getActivateInfo(profile); | ||||
|     utility::string_t payload =  utility::conversions::to_string_t(activationInfo); | ||||
|  | ||||
|     request[U("payload")] = json::value::string(payload); | ||||
|  | ||||
|     return utility::conversions::to_utf8string(request.serialize()); | ||||
| } | ||||
|  | ||||
| string createResponse(string payload) | ||||
| { | ||||
|     // Activation parameters | ||||
|     json::value response; | ||||
|  | ||||
|     // placeholder stuff; will likely change | ||||
|     utility::string_t tmp = utility::conversions::convertstring("response"); | ||||
|     response[U("method")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("key"); | ||||
|     response[U("apiKey")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring(PROJECT_VER); | ||||
|     response[U("appVersion")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring(PROTOCOL_VERSION); | ||||
|     response[U("protocolVersion")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("ok"); | ||||
|     response[U("status")] = json::value::string(tmp); | ||||
|  | ||||
|     tmp = utility::conversions::convertstring("all\'s good!"); | ||||
|     response[U("message")] = json::value::string(tmp); | ||||
|  | ||||
|     // payload  | ||||
|     tmp = utility::conversions::convertstring( encodeBase64(payload) ); | ||||
|     response[U("payload")] = json::value::string(tmp); | ||||
|  | ||||
|     return utility::conversions::to_utf8string(response.serialize()); | ||||
| } | ||||
|  | ||||
| void dumpMessage(utility::string_t tmp) | ||||
| { | ||||
|     web::json::value parsed = web::json::value::parse(tmp); | ||||
|  | ||||
|     if ( !parsed.has_field(U("method"))          || !parsed.has_field(U("apiKey")) || !parsed.has_field(U("appVersion"))  ||  | ||||
|          !parsed.has_field(U("protocolVersion")) || !parsed.has_field(U("status")) || !parsed.has_field(U("message"))     || | ||||
|          !parsed.has_field(U("payload"))  ) { | ||||
|             cout << "error: dumpMessage message is empty" << endl; | ||||
|             return; | ||||
|     } | ||||
|  | ||||
|     utility::string_t out = parsed[U("method")].as_string(); | ||||
|     cout << "method: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("apiKey")].as_string(); | ||||
|     cout << "apiKey: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("appVersion")].as_string(); | ||||
|     cout << "appVersion: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("protocolVersion")].as_string(); | ||||
|     cout << "protocolVersion: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("status")].as_string(); | ||||
|     cout << "status: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("message")].as_string(); | ||||
|     cout << "message: " << utility::conversions::to_utf8string(out) << endl; | ||||
|  | ||||
|     out = parsed[U("payload")].as_string(); | ||||
|     cout << "payload: " << utility::conversions::to_utf8string(out) << endl; | ||||
| } | ||||
							
								
								
									
										48
									
								
								commands.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								commands.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| /* | ||||
| Copyright 2019 Intel Corporation | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| 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. | ||||
| */ | ||||
|  | ||||
| #ifndef __COMMANDS_H__ | ||||
| #define __COMMANDS_H__ | ||||
|  | ||||
| #include <iostream> | ||||
| #include <string> | ||||
|  | ||||
| #include <cpprest/ws_client.h> | ||||
| #include <cpprest/json.h> | ||||
| #include <cpprest/streams.h> | ||||
|  | ||||
| using namespace std; | ||||
| using namespace web::websockets::client; | ||||
| using namespace web; | ||||
|  | ||||
| #define PROTOCOL_VERSION "2.0.0" | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| #define convertstring   to_utf16string | ||||
| #else | ||||
| #define convertstring   to_utf8string | ||||
| #endif | ||||
|  | ||||
| string getDNSInfo(); | ||||
| string createActivationRequest(string profile); | ||||
| json::value getCertificateHashes(); | ||||
| string createResponse(string payload); | ||||
| string getActivateInfo(string profile); | ||||
| string encodeBase64(string str); | ||||
| string decodeBase64(string str); | ||||
| void dumpMessage(string tmp); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										15
									
								
								lms.cmake.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								lms.cmake.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| cmake_minimum_required(VERSION 3.1) | ||||
|  | ||||
| project(lms-download NONE) | ||||
|  | ||||
| include(ExternalProject) | ||||
| ExternalProject_Add(lms | ||||
|   GIT_REPOSITORY    https://github.com/intel/lms.git | ||||
|   GIT_TAG           v1932.0.0.0 | ||||
|   SOURCE_DIR        "${CMAKE_BINARY_DIR}/lms-src" | ||||
|   BINARY_DIR        "${CMAKE_BINARY_DIR}/lms-build" | ||||
|   CONFIGURE_COMMAND "" | ||||
|   BUILD_COMMAND     "" | ||||
|   INSTALL_COMMAND   "" | ||||
|   TEST_COMMAND      "" | ||||
| ) | ||||
							
								
								
									
										88
									
								
								lms.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								lms.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| /* | ||||
| Copyright 2019 Intel Corporation | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| 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 "lms.h" | ||||
| #include <string.h> | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| // Windows | ||||
| #include <Ws2tcpip.h> | ||||
| #else | ||||
| // Linux | ||||
| #include <sys/types.h> | ||||
| #include <sys/socket.h> | ||||
| #include <netdb.h> | ||||
| #endif | ||||
|  | ||||
| SOCKET lmsConnect() | ||||
| { | ||||
|     std::string lmsAddress = "localhost"; | ||||
|     std::string lmsPort = "16992"; | ||||
|     SOCKET s = INVALID_SOCKET; | ||||
|     struct addrinfo *addr, hints; | ||||
|  | ||||
| #ifdef _WIN32 | ||||
|     WSADATA wsa; | ||||
|     if (WSAStartup(MAKEWORD(2, 2), &wsa) != 0) | ||||
|     { | ||||
|         throw std::runtime_error("error: unable to connect to LMS"); | ||||
|     } | ||||
| #endif | ||||
|  | ||||
|     memset(&hints, 0, sizeof(hints)); | ||||
|      | ||||
|     hints.ai_family = AF_UNSPEC; | ||||
|     hints.ai_socktype = SOCK_STREAM; | ||||
|     hints.ai_protocol = IPPROTO_TCP; | ||||
|  | ||||
|     if (getaddrinfo(lmsAddress.c_str(), lmsPort.c_str(), &hints, &addr) != 0) | ||||
|     { | ||||
|         throw std::runtime_error("error: unable to connect to LMS"); | ||||
|     } | ||||
|  | ||||
|     if (addr == NULL) | ||||
|     { | ||||
|         throw std::runtime_error("error: unable to connect to LMS"); | ||||
|     } | ||||
|  | ||||
|     for (addr; addr != NULL; addr = addr->ai_next) | ||||
|     { | ||||
|         s = socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol); | ||||
|         if (s == INVALID_SOCKET) | ||||
|         { | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (connect(s, addr->ai_addr, (int)addr->ai_addrlen) == 0) | ||||
|         { | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         closesocket(s); | ||||
|         s = INVALID_SOCKET; | ||||
|     } | ||||
|  | ||||
|     if (addr != NULL) | ||||
|     { | ||||
|         freeaddrinfo(addr); | ||||
|     } | ||||
|  | ||||
|     if (s == INVALID_SOCKET) | ||||
|     { | ||||
|         throw std::runtime_error("error: unable to connect to LMS"); | ||||
|     } | ||||
|  | ||||
|     return s; | ||||
| } | ||||
							
								
								
									
										44
									
								
								lms.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								lms.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| /* | ||||
| Copyright 2019 Intel Corporation | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| 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. | ||||
| */ | ||||
|  | ||||
| #ifndef __LMS_H__ | ||||
| #define __LMS_H__ | ||||
|  | ||||
| #include <iostream> | ||||
| #include <string> | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| #include <winsock2.h> | ||||
| #else | ||||
| #include <unistd.h> | ||||
| typedef int SOCKET; | ||||
| #define INVALID_SOCKET (SOCKET)(-1) | ||||
| #endif | ||||
|  | ||||
| #ifdef _WIN32 | ||||
| // Windows | ||||
| #else | ||||
| // Linux | ||||
| static inline int closesocket(int fd) | ||||
| { | ||||
|     return close(fd); | ||||
| } | ||||
| #define SD_BOTH SHUT_RDWR | ||||
| #endif | ||||
|  | ||||
| SOCKET lmsConnect(); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										480
									
								
								main.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										480
									
								
								main.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,480 @@ | ||||
| /* | ||||
| Copyright 2019 Intel Corporation | ||||
|  | ||||
| Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| you may not use this file except in compliance with the License. | ||||
| You may obtain a copy of the License at | ||||
|  | ||||
|     http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
| 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 <cpprest/ws_client.h> | ||||
| #include <cpprest/streams.h> | ||||
| #include <iostream> | ||||
| #include <string> | ||||
| #include <thread> | ||||
| #include <boost/algorithm/string.hpp> | ||||
| #include "commands.h" | ||||
| #include "lms.h" | ||||
| #include "version.h" | ||||
|  | ||||
| using namespace std; | ||||
| using namespace web; | ||||
| using namespace web::websockets::client; | ||||
|  | ||||
| #include <cpprest/json.h> | ||||
|  | ||||
| void showUsage(); | ||||
|  | ||||
| string websocket_address  = ""; | ||||
| string server_profile = ""; | ||||
| string websocket_proxy = ""; | ||||
|  | ||||
| long long timeoutTimer = 0; | ||||
| const int MAX_TIMEOUT = 10; // seconds | ||||
| bool timeoutThreadAlive = true; | ||||
|  | ||||
| void timeoutFunc(std::condition_variable *cv, std::mutex *mx) | ||||
| { | ||||
|     while (timeoutThreadAlive)  | ||||
|     { | ||||
|         std::chrono::time_point<std::chrono::system_clock> now = std::chrono::system_clock::now(); | ||||
|         auto duration = now.time_since_epoch(); | ||||
|         long long currTime = std::chrono::duration_cast<std::chrono::seconds>(duration).count(); | ||||
|  | ||||
|         if (currTime > timeoutTimer)  | ||||
|         { | ||||
|             if (currTime - timeoutTimer >= MAX_TIMEOUT)  | ||||
|             { | ||||
|                 cv->notify_all(); | ||||
|  | ||||
|                 // check if timeoutTimer is not 0 since we explicitly set to zero when an | ||||
|                 // activation is successfull. If it's not zero, we are in a time out scenario. | ||||
|                 if (timeoutTimer) | ||||
|                 { | ||||
|                     cout << endl << "Timed-out due to inactivity." <<endl; | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|             std::this_thread::sleep_for(std::chrono::seconds(1)); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| int main(int argc, char *argv[]) | ||||
| { | ||||
|     string activationInfo; | ||||
|  | ||||
|     bool gotURL = false; | ||||
|     bool gotProfile = false; | ||||
|     bool gotProxy = false; | ||||
|     bool gotDns = false; | ||||
|      | ||||
|     if (argc==1) | ||||
|     { | ||||
|         std::cout << "Use --h, --help for help." << std::endl; | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     for (int i=1; i<argc; i++) | ||||
|     { | ||||
|         if ( (boost::equals(argv[i], "--help")) || (boost::equals(argv[i], "--h"))  ) | ||||
|         { | ||||
|             showUsage(); | ||||
|             return 0; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     for (int i=1; i<argc; i++) | ||||
|     { | ||||
|         if ( (boost::equals(argv[i], "--url"))  || (boost::equals(argv[i], "--u")) ) | ||||
|         { | ||||
|             if (i+1<argc)  | ||||
|             { | ||||
|                 gotURL = true; | ||||
|                 websocket_address = argv[++i]; | ||||
|             } | ||||
|         }  | ||||
|         else if ( (boost::equals(argv[i], "--profile")) || (boost::equals(argv[i], "--p")) ) | ||||
|         { | ||||
|             if (i+1<argc)  | ||||
|             { | ||||
|                 gotProfile = true; | ||||
|                 server_profile = argv[++i]; | ||||
|             } | ||||
|         } | ||||
|         else if ( (boost::equals(argv[i], "--proxy")) ||(boost::equals(argv[i], "--x")) ) | ||||
|         { | ||||
|             if (i+1<argc)  | ||||
|             { | ||||
|                 gotProxy = true; | ||||
|                 websocket_proxy = argv[++i]; | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             std::cout << "Unrecognized command line arguments." << std::endl; | ||||
|             std::cout << "Use --h, --help for help." << std::endl; | ||||
|             return 0; | ||||
|         } | ||||
|          | ||||
|     } | ||||
|  | ||||
|     if (!gotURL || !gotProfile) | ||||
|     { | ||||
|         std::cout << "Incorrect or missing arguments." << std::endl; | ||||
|         std::cout << "Use --h, --help for help." << std::endl; | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     // Print version info | ||||
|     cout << PROJECT_NAME << " v" PROJECT_VER << endl; | ||||
|  | ||||
|     try { | ||||
|         // Get activation info | ||||
|         activationInfo = createActivationRequest(server_profile); | ||||
|     } | ||||
|     catch (...) | ||||
|     { | ||||
|         std::cerr << endl << "Unable to get activation info. Check AMT configuration." << endl; | ||||
|         return 1; | ||||
|     } | ||||
|  | ||||
|     try | ||||
|     { | ||||
|         // Check if LMS is available | ||||
|         SOCKET s = lmsConnect(); | ||||
|         closesocket(s); | ||||
|     } | ||||
|     catch (...) | ||||
|     { | ||||
|         std::cerr << endl << "Unable to connect to Local Management Service (LMS). Please ensure LMS is running." << endl; | ||||
|         return 1; | ||||
|     } | ||||
|  | ||||
|     // Show activation info | ||||
| #ifdef DEBUG | ||||
|     cout << "Activation info: " << endl << activationInfo << endl; | ||||
| #endif | ||||
|  | ||||
|     // WebSocket Interface | ||||
|     websocket_client_config client_config; | ||||
|     if (!websocket_proxy.empty()) | ||||
|     { | ||||
|         client_config.set_proxy(web::web_proxy(utility::conversions::to_string_t(websocket_proxy))); | ||||
|     } | ||||
| #ifdef DEBUG | ||||
|     // skip certificate verification if debug build | ||||
|     cout << "!!! SKIPPING CERTIFICATE VERIFICATION !!!" << endl; | ||||
|     client_config.set_validate_certificates(false); | ||||
| #endif | ||||
|     websocket_callback_client client(client_config); | ||||
|     std::condition_variable cv; | ||||
|     std::mutex mx; | ||||
|     SOCKET s; | ||||
|  | ||||
|     // set receive handler | ||||
|     client.set_message_handler([&client, &mx, &cv, &s](websocket_incoming_message ret_msg)  | ||||
|     { | ||||
|         // kick the timer | ||||
|         std::chrono::time_point<std::chrono::system_clock> now = std::chrono::system_clock::now(); | ||||
|         auto duration = now.time_since_epoch(); | ||||
|         timeoutTimer = std::chrono::duration_cast<std::chrono::seconds>(duration).count(); | ||||
|  | ||||
|         try | ||||
|         { | ||||
|             // handle message from server... | ||||
|             string rcv_websocket_msg = ret_msg.extract_string().get(); | ||||
| #ifdef DEBUG | ||||
|             cout << endl << "<<<<< Received Message " << endl; | ||||
|             cout << rcv_websocket_msg << endl; | ||||
| #endif | ||||
|             cout << "." << std::flush; // dot status output | ||||
|  | ||||
|             // parse incoming JSON message | ||||
|             utility::string_t tmp = utility::conversions::convertstring(rcv_websocket_msg); | ||||
|             web::json::value parsed = web::json::value::parse(tmp); | ||||
|  | ||||
|             utility::string_t out = U(""); | ||||
|             string msgMethod = ""; | ||||
|             string msgApiKey = ""; | ||||
|             string msgAppVersion = ""; | ||||
|             string msgProtocolVersion = ""; | ||||
|             string msgStatus = ""; | ||||
|             string msgMessage = ""; | ||||
|             string msgPayload = ""; | ||||
|             string payloadDecoded = ""; | ||||
|  | ||||
|             if ( !parsed.has_field(U("method"))          || !parsed.has_field(U("apiKey")) || !parsed.has_field(U("appVersion"))  ||  | ||||
|                  !parsed.has_field(U("protocolVersion")) || !parsed.has_field(U("status")) || !parsed.has_field(U("message"))     || | ||||
|                  !parsed.has_field(U("payload"))  ) { | ||||
|                      std::cerr << endl << "Received incorrectly formatted message." << endl; | ||||
|                      cv.notify_all(); | ||||
|                      timeoutThreadAlive = false; | ||||
|                      return; | ||||
|             } | ||||
|  | ||||
|             try  | ||||
|             { | ||||
|                 out = parsed[U("method")].as_string(); | ||||
|                 msgMethod = utility::conversions::to_utf8string(out); | ||||
|  | ||||
|                 out = parsed[U("apiKey")].as_string(); | ||||
|                 msgApiKey = utility::conversions::to_utf8string(out); | ||||
|  | ||||
|                 out = parsed[U("appVersion")].as_string(); | ||||
|                 msgAppVersion = utility::conversions::to_utf8string(out); | ||||
|  | ||||
|                 out = parsed[U("protocolVersion")].as_string(); | ||||
|                 msgProtocolVersion = utility::conversions::to_utf8string(out); | ||||
|  | ||||
|                 out = parsed[U("status")].as_string(); | ||||
|                 msgStatus = utility::conversions::to_utf8string(out); | ||||
|  | ||||
|                 out = parsed[U("message")].as_string(); | ||||
|                 msgMessage = utility::conversions::to_utf8string(out); | ||||
|             } | ||||
|             catch (...) | ||||
|             { | ||||
|                 std::cerr << endl << "Received message parse error." << endl; | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|  | ||||
| #ifdef DEBUG | ||||
|             cout << msgMethod << ", " << msgStatus << ", " << msgMessage << endl; | ||||
|             cout << rcv_websocket_msg << endl; | ||||
| #endif | ||||
|  | ||||
|             // process any messages we can | ||||
|             //   - if success, done | ||||
|             //   - if error, get out | ||||
|             if (boost::iequals(msgMethod, "success")) | ||||
|             { | ||||
|                 // cleanup | ||||
|                 timeoutTimer = 0; | ||||
|  | ||||
|                 // exit | ||||
|                 cout << endl << msgMessage << endl; | ||||
|                 return; | ||||
|             } | ||||
|             else if (boost::iequals(msgMethod, "error")) | ||||
|             { | ||||
|                 // cleanup | ||||
|                 timeoutTimer = 0; | ||||
|  | ||||
|                 // exit | ||||
|                 cout << endl << msgMessage << endl; | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             // process payload afterward since success/error messages have zero length | ||||
|             // payloads which cause an exception to be thrown | ||||
|             try  | ||||
|             { | ||||
|                 out = parsed[U("payload")].as_string(); | ||||
|  | ||||
|                 if (out.length()>0) | ||||
|                 { | ||||
|                     msgPayload = utility::conversions::to_utf8string(out); | ||||
|                      | ||||
|                     // decode payload | ||||
|                     payloadDecoded = decodeBase64(msgPayload); | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     // no payload, nothing to do | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|             catch (...) | ||||
|             { | ||||
|                 std::cerr << endl << "JSON format error. Unable to parse message." << endl; | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             try | ||||
|             { | ||||
|                 // conntect to lms | ||||
|                 s = lmsConnect(); | ||||
|             } | ||||
|             catch (...) | ||||
|             { | ||||
|                 std::cerr << endl << "Unable to connect to Local Management Service (LMS). Please ensure LMS is running." << endl; | ||||
|                 return; | ||||
|             } | ||||
|              | ||||
| #ifdef DEBUG | ||||
|         cout << endl << "vvvvv Sending Message " << endl; | ||||
|         cout << payloadDecoded << endl;         | ||||
| #endif         | ||||
|  | ||||
|             // send message to LMS | ||||
|             if (send(s, payloadDecoded.c_str(), (int)payloadDecoded.length(), 0) < 0) | ||||
|             { | ||||
|                 throw std::runtime_error("error: socket send"); | ||||
|             } | ||||
|  | ||||
|             // handle response message from LMS | ||||
|             int fd = ((int) s) + 1; | ||||
|             fd_set rset; | ||||
|             FD_ZERO(&rset); | ||||
|             FD_SET(s, &rset); | ||||
|  | ||||
|             timeval timeout; | ||||
|             memset(&timeout, 0, sizeof(timeval)); | ||||
|             timeout.tv_sec = 2;  | ||||
|             timeout.tv_usec = 0; | ||||
|  | ||||
|             // read until connection is closed by LMS | ||||
|             while (1) | ||||
|             { | ||||
|                 string superBuffer = ""; | ||||
|                 while (1) | ||||
|                 { | ||||
|                     int res = select(fd, &rset, NULL, NULL, &timeout); | ||||
|                     if (res == 0)  | ||||
|                     { | ||||
|                         // we timed-out waiting for the ME. ME usually delivers data very fast. If | ||||
|                         // we time out, it means that there is no more data that the ME needs to send, | ||||
|                         // but it's keeping the connection open. | ||||
|                         break; | ||||
|                     } | ||||
|  | ||||
|                     // read from LMS | ||||
|                     char recv_buffer[4096]; | ||||
|                     memset(recv_buffer, 0, 4096); | ||||
|                     res = recv(s, recv_buffer, 4096, 0); | ||||
|                     if (res > 0) | ||||
|                     { | ||||
| #ifdef DEBUG | ||||
|                         cout << endl << "^^^^^ Received Message" << endl; | ||||
|                         cout << recv_buffer << endl; | ||||
| #endif | ||||
|                         superBuffer += recv_buffer; | ||||
|                     } | ||||
|                     else if (res < 0) | ||||
|                     { | ||||
|                         // on LMS read exception | ||||
|                         break; | ||||
|                     } | ||||
|                     else | ||||
|                     { | ||||
|                         // case where res is zero bytes | ||||
|                         // discussion below, but select returns 1 with recv returning 0 to indicate close | ||||
|                         // https://stackoverflow.com/questions/2992547/waiting-for-data-via-select-not-working | ||||
|                         break; | ||||
|                     } | ||||
|                 } // while select() | ||||
|  | ||||
|                 // if there is some data send it | ||||
|                 if (superBuffer.length() > 0)  | ||||
|                 { | ||||
|                     string response = createResponse(superBuffer.c_str()); | ||||
|                     websocket_outgoing_message send_websocket_msg; | ||||
|                     string send_websocket_buffer(response); | ||||
|                     send_websocket_msg.set_utf8_message(send_websocket_buffer); | ||||
| #ifdef DEBUG | ||||
|                     cout << endl << ">>>>> Sending Message" << endl; | ||||
|                     cout << superBuffer << endl; | ||||
|                     cout << send_websocket_buffer << endl; | ||||
| #endif | ||||
|                     client.send(send_websocket_msg).wait(); | ||||
|  | ||||
|                     // done | ||||
|                     closesocket(s); | ||||
|                     return; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             closesocket(s); | ||||
|         } | ||||
|         catch (...) | ||||
|         { | ||||
|             std::cerr << endl << "Communication error in receive handler." << endl; | ||||
|             closesocket(s); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // set close handler | ||||
|     client.set_close_handler([&mx,&cv](websocket_close_status status, const utility::string_t &reason, const std::error_code &code)  | ||||
|     { | ||||
|         // websocket closed by server, notify main thread  | ||||
|         cv.notify_all(); | ||||
|     }); | ||||
|  | ||||
|     try | ||||
|     { | ||||
|         // Connect to web socket server; AMT activation server | ||||
|         client.connect(utility::conversions::to_string_t(websocket_address)).wait(); | ||||
|     } | ||||
|     catch (...) | ||||
|     { | ||||
|         std::cerr << "Unable to connect to websocket server. Please check url." << endl; | ||||
|         exit(1); | ||||
|     } | ||||
|  | ||||
|     try  | ||||
|     { | ||||
|         // Send activationParams to websocket | ||||
|         websocket_outgoing_message out_msg; | ||||
|         out_msg.set_utf8_message(activationInfo); | ||||
|          | ||||
| #ifdef DEBUG | ||||
|         cout << endl << ">>>>> Sending Activiation Info" << endl; | ||||
|         cout << activationInfo << endl; | ||||
| #endif         | ||||
|         client.send(out_msg).wait(); | ||||
|     } | ||||
|     catch (...) | ||||
|     { | ||||
|         std::cerr << endl << "Unable to send message to websocket server." << endl; | ||||
|         exit(1); | ||||
|     } | ||||
|  | ||||
|     std::chrono::time_point<std::chrono::system_clock> now = std::chrono::system_clock::now(); | ||||
|     auto duration = now.time_since_epoch(); | ||||
|     timeoutTimer = std::chrono::duration_cast<std::chrono::seconds>(duration).count(); | ||||
|     std::thread timeoutThread(timeoutFunc, &cv, &mx); | ||||
|  | ||||
|     // wait for server to send success/failure command | ||||
|     std::unique_lock<std::mutex> lock(mx); | ||||
|     cv.wait(lock); | ||||
|  | ||||
|     // wait for timeout thread | ||||
|     timeoutThread.join(); | ||||
|  | ||||
|     // clean-up websocket | ||||
|     client.close().wait(); | ||||
|  | ||||
|     // clean-up socket | ||||
|     if (s) { | ||||
|         shutdown(s, SD_BOTH); | ||||
|         closesocket(s); | ||||
|     } | ||||
|  | ||||
|     exit(0); | ||||
| } | ||||
|  | ||||
| void showUsage() | ||||
| { | ||||
|     cout << "Usage: " << PROJECT_NAME << " --url <url> --profile <name> [--proxy <addr>]" << endl; | ||||
|     cout << "Required:" << endl; | ||||
|     cout << "  --u, --url <url>                 websocket server" << endl; | ||||
|     cout << "  --p, --profile <name>            server profile" << endl; | ||||
|     cout << "Optional:" << endl; | ||||
|     cout << "  --x, --proxy <addr>              proxy address and port" << endl; | ||||
|     cout << endl; | ||||
|     cout << "Examples:" << endl; | ||||
|     cout << "  " << PROJECT_NAME << " --url wss://localhost --profile profile1" << endl; | ||||
|     cout << "  " << PROJECT_NAME << " --u wss://localhost --profile profile1 --proxy http://proxy.com:1000" << endl; | ||||
|     cout << "  " << PROJECT_NAME << " --u wss://localhost --p profile1 --x http://proxy.com:1000" << endl; | ||||
|     cout << endl; | ||||
| } | ||||
							
								
								
									
										10
									
								
								version.h.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								version.h.in
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| #ifndef __VERSION_H__ | ||||
| #define __VERSION_H__ | ||||
|  | ||||
| #define PROJECT_NAME "@PROJECT_NAME@" | ||||
| #define PROJECT_VER  "@PROJECT_VERSION@" | ||||
| #define PROJECT_VER_MAJOR "@PROJECT_VERSION_MAJOR@" | ||||
| #define PROJECT_VER_MINOR "@PROJECT_VERSION_MINOR@" | ||||
| #define PTOJECT_VER_PATCH "@PROJECT_VERSION_PATCH@" | ||||
|  | ||||
| #endif // __VERSION_H__ | ||||
		Reference in New Issue
	
	Block a user