#include "epm_handler_common.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "tc/envelope.h" #include #include "ae/dataset.h" #include #include #include #include #include #include #include #include "ce/ce.h" #include #include #include #include #include #include "string" #include "sstream" #include #include #include #include
#include "ctime" #include "tchar.h" #include "jni.h" #ifdef WIN32 #include #include #else #include #include #endif #include #include #define MAX_PATH_LEN 256 #define GUID_LEN 64 #ifdef WIN32 #define ACCESS(fileName,accessMode) _access(fileName,accessMode) #define MKDIR(path) _mkdir(path) #else #define ACCESS(fileName,accessMode) access(fileName,accessMode) #define MKDIR(path) mkdir(path,S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) #endif #define LOCALJARPATH "D:\\Siemens\\Teamcenter12\\bin\\tc_extra_jar\\" #define REMOTEJARPATH "D:\\app\\Siemens\\Teamcenter12\\bin\\tc_extra_jar\\" using namespace std; extern "C" int POM_AM__set_application_bypass(logical bypass); int32_t createDirectory(const std::string &directoryPath) { uint32_t dirPathLen = directoryPath.length(); if (dirPathLen > MAX_PATH_LEN) { return -1; } char tmpDirPath[MAX_PATH_LEN] = { 0 }; for (uint32_t i = 0; i < dirPathLen; ++i) { tmpDirPath[i] = directoryPath[i]; if (tmpDirPath[i] == '\\' || tmpDirPath[i] == '/') { if (ACCESS(tmpDirPath, 0) != 0) { int32_t ret = MKDIR(tmpDirPath); if (ret != 0) { return ret; } } } } return 0; } static void replace_dataset_named_reference(tag_t dataset_tag, tag_t old_file, const char *new_file_path, char *ref_name) { IMF_file_data_p_t file_data; int index = 0; char *name; IMF_get_file_access(old_file, 0, &file_data); tag_t new_file_tag = NULLTAG; AOM_lock(old_file); IMF_ask_original_file_name2(old_file, &name); printf("old name:%s\n",name); IMF_replace_file_and_get_new_tag(old_file, new_file_path, FALSE, &new_file_tag); AOM_lock(dataset_tag); AE_replace_dataset_named_ref2(dataset_tag, old_file, ref_name, AE_PART_OF, new_file_tag); AE_save_myself(dataset_tag); IMF_set_original_file_name2(new_file_tag, name); AOM_unlock(dataset_tag); IMF_release_file_access(&file_data); } bool isTypeOf(tag_t objtag, const char * type_name) { tag_t type = NULLTAG; TCTYPE_ask_object_type(objtag, &type); tag_t item_type = NULLTAG; TCTYPE_find_type(type_name, "", &item_type); bool is_type = false; if (item_type != NULLTAG) { logical isok = FALSE; TCTYPE_is_type_of(type, item_type, &isok); if (isok) { is_type = true; } } return is_type; } string GBKToUTF8(const char* strGBK) { int len = MultiByteToWideChar(CP_ACP, 0, strGBK, -1, NULL, 0); wchar_t* wstr = new wchar_t[len + 1]; memset(wstr, 0, len + 1); MultiByteToWideChar(CP_ACP, 0, strGBK, -1, wstr, len); len = WideCharToMultiByte(CP_UTF8, 0, wstr, -1, NULL, 0, NULL, NULL); char* str = new char[len + 1]; memset(str, 0, len + 1); WideCharToMultiByte(CP_UTF8, 0, wstr, -1, str, len, NULL, NULL); string strTemp = str; if (wstr) delete[] wstr; if (str) delete[] str; return strTemp; } /* 映射 -path 路径 -name 用户名 */ static int sign(JNIEnv* env, jclass class_Test, const char* path, char* name, char* sign_location, char* sign_str) { stringstream ss; printf("signin\n"); jmethodID methodId_main = env->GetStaticMethodID(class_Test, "sign", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)I"); if (methodId_main == NULL) { printf("error, check jar location\n"); ss << "找不到sign2fileRunnable.jar 确认路径为:"; ss << REMOTEJARPATH; EMH_store_error_s1(EMH_severity_error, EMH_ATTR_error_base, ss.str().c_str()); return 0; } string path1 = GBKToUTF8(path); string name1 = GBKToUTF8(name); string sign_location1 = GBKToUTF8(sign_location); string sign_str1 = GBKToUTF8(sign_str); jstring jpath = env->NewStringUTF(path1.c_str()); jstring jname = env->NewStringUTF(name1.c_str()); jstring jsign_location = env->NewStringUTF(sign_location1.c_str()); jstring jsign_str = env->NewStringUTF(sign_str1.c_str()); printf("path:%s\nname:%s\nsign_location:%s\nsign_str:%s\n", path1.c_str(), name1.c_str(), sign_location1.c_str(), sign_str1.c_str()); jint result = 0; result = env->CallStaticIntMethod(class_Test, methodId_main, jpath, jname, jsign_location, jsign_str); int res = result; printf("result:%d\n", res); printf("signout\n"); return res; } void add_path(JNIEnv* env, const std::string& path) { const std::string urlPath = "file:/" + path; jclass classLoaderCls = env->FindClass("java/lang/ClassLoader"); printf("111\n"); jmethodID getSystemClassLoaderMethod = env->GetStaticMethodID(classLoaderCls, "getSystemClassLoader", "()Ljava/lang/ClassLoader;"); jobject classLoaderInstance = env->CallStaticObjectMethod(classLoaderCls, getSystemClassLoaderMethod); jclass urlClassLoaderCls = env->FindClass("java/net/URLClassLoader"); jmethodID addUrlMethod = env->GetMethodID(urlClassLoaderCls, "addURL", "(Ljava/net/URL;)V"); jclass urlCls = env->FindClass("java/net/URL"); jmethodID urlConstructor = env->GetMethodID(urlCls, "", "(Ljava/lang/String;)V"); jobject urlInstance = env->NewObject(urlCls, urlConstructor, env->NewStringUTF(urlPath.c_str())); env->CallVoidMethod(classLoaderInstance, addUrlMethod, urlInstance); std::cout << "Added " << urlPath << " to the classpath." << std::endl; } bool find(int prop_cnt, char** props, char* prop) { for (int i = 0;i < prop_cnt;i++) { if (tc_strcmp(props[i], prop) == 0) { return true; } } return false; } //吉德表单签名 int jd_signoff(EPM_action_message_t msg) { const char *sign_location, *sign_str; int ifail = ITK_ok, att_count, objects_count, pref_count, prop_cnt; tag_t rootTask, *attachments, user_tag, *objects; char *user_name, **pref_values, **prop_names; logical verdict; bool flag; stringstream sign_date, errorMsg; //获取时间 time_t t; time(&t); tm* time = localtime(&t); stringstream date; date << time->tm_year + 1900 << "."; date << time->tm_mon + 1 << "."; date << time->tm_mday; //获取guid //获取guid char buffer[GUID_LEN] = { 0 }; GUID guid; if (CoCreateGuid(&guid)) { fprintf(stderr, "create guid error\n"); return NULL; } _snprintf(buffer, sizeof(buffer), "%08X-%04X-%04x-%02X%02X-%02X%02X%02X%02X%02X%02X", guid.Data1, guid.Data2, guid.Data3, guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3], guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]); printf("guid: %s\n", buffer); //获取数据集 char *temp_path, *attach_type; int n_attachs; tag_t ref_obj, excelx_tag, excel_tag, wordx_tag, word_tag, type_tag, word_ref, excel_ref, *dataset; ITEM_attached_object_t *rev_attachments; AE_reference_type_t ae_ref; TCTYPE_find_type("MSExcelX", "Dataset", &excelx_tag); TCTYPE_find_type("MSExcel", "Dataset", &excel_tag); TCTYPE_find_type("MSWord", "Dataset", &word_tag); TCTYPE_find_type("MSWordX", "Dataset", &wordx_tag); temp_path = getenv("temp"); POM_AM__set_application_bypass(true); //获取当前用户 POM_get_user(&user_name, &user_tag); //获取选中的目标对象 TC_argument_list_t * arguments = msg.arguments; int arg_cnt = TC_number_of_arguments(arguments), status = 0; EPM_ask_root_task(msg.task, &rootTask); EPM_ask_attachments(rootTask, EPM_target_attachment, &att_count, &attachments); //获取流程中配置的handler属性 map paras; for (auto i = 0; i < arg_cnt; i++) { char *temp_key, *temp_val; ITK_ask_argument_named_value(TC_next_argument(arguments), &temp_key, &temp_val); paras[temp_key] = temp_val; } sign_location = paras["sign_location"].c_str(); printf("sign_location:%s\n", sign_location); sign_date << sign_location; sign_date << "rq"; //读取首选项 PREF_ask_char_values("jd2_signoff_list", &pref_count, &pref_values); flag = false; stringstream temps; for (int i = 0;i < pref_count;i++) { printf("signoff:%s\n", pref_values[i]); temps << pref_values[i]; temps << ","; if (strcmp(pref_values[i], sign_location) == 0) { flag = true; } } if (strcmp("clear", sign_location) == 0) { flag = true; } if (!flag) { errorMsg << "流程参数\""; errorMsg << sign_location; errorMsg << "\"不存在于首选项中,检查首选项\"jd2_signoff_list\"!"; EMH_store_error_s1(EMH_severity_error, EMH_ATTR_error_base, errorMsg.str().c_str()); return -1; } //创建tc_export目录 stringstream dir; dir << temp_path; dir << "\\tc_export\\"; createDirectory(dir.str()); LPVOID lpMsgBuf; FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language (LPTSTR)&lpMsgBuf, 0, NULL ); //申请虚拟机 //wstringstream wss; //wss << getenv("JRE_HOME") << "\\bin\\server\\jvm.dll"; //JavaVMInitArgs vmArgs; //vmArgs.version = JNI_VERSION_1_8; //const int OPTION_COUNT = 2; //vmArgs.nOptions = OPTION_COUNT; //JavaVMOption options[OPTION_COUNT] = { 0 }; //options[0].optionString = "-Djava.class.path=D:\\Siemens\\Teamcenter11\\bin\\tc_extra_jar\\sign2fileRunnable.jar"; //options[1].optionString = "-Xmx1024m"; //vmArgs.options = options; //vmArgs.ignoreUnrecognized = JNI_TRUE; //printf("before load dll\n"); //HMODULE hModule = LoadLibrary(wss.str().c_str()); //if (hModule == NULL) { // printf("error load dll"); // return -1; //} //printf("after load dll\n"); //typedef jint(JNICALL *CreateJavaVMFuncPtr)(JavaVM **pvm, void **penv, void *args); //CreateJavaVMFuncPtr CreateJavaVM = (CreateJavaVMFuncPtr)GetProcAddress(hModule, "JNI_CreateJavaVM"); //JavaVM *jvm = nullptr; //JNIEnv *env = nullptr; //jint res = (*CreateJavaVM)(&jvm, (void**)&env, &vmArgs); //if (res < 0) { // printf("error load jvm"); // return -1; //} //printf("after load jvm\n"); wstringstream wss; wss << getenv("JRE_HOME") << "\\bin\\server\\jvm.dll"; JavaVMInitArgs vmArgs; const int OPTION_COUNT = 1; vmArgs.nOptions = OPTION_COUNT; JavaVMOption options[OPTION_COUNT] = { 0 }; options[0].optionString = "-Xmx1024m"; vmArgs.options = options; vmArgs.version = JNI_VERSION_1_8; vmArgs.ignoreUnrecognized = JNI_TRUE; printf("before load dll\n"); HMODULE hModule = LoadLibrary(wss.str().c_str()); if (hModule == NULL) { printf("error load dll"); return -1; } printf("after load dll\n"); typedef jint(JNICALL *CreateJavaVMFuncPtr)(JavaVM **pvm, void **penv, void *args); CreateJavaVMFuncPtr CreateJavaVM = (CreateJavaVMFuncPtr)GetProcAddress(hModule, "JNI_CreateJavaVM"); typedef jint(JNICALL * GetCreatedJavaVMs)(JavaVM**, jsize, jsize*); GetCreatedJavaVMs jni_GetCreatedJavaVMs; jni_GetCreatedJavaVMs = (GetCreatedJavaVMs)GetProcAddress(GetModuleHandle( TEXT("jvm.dll")), "JNI_GetCreatedJavaVMs"); JavaVM *jvm = nullptr, *jvms; JNIEnv *env = nullptr; jsize size = 0; jni_GetCreatedJavaVMs(&jvms, 0, &size); printf("before load jvm\n"); if (size) { JavaVM** buffer = new JavaVM*[size]; jni_GetCreatedJavaVMs(buffer, size, &size); buffer[0]->GetEnv((void**)&env, JNI_VERSION_1_8); jint result = buffer[0]->AttachCurrentThread((void**)&env, NULL); } else { jint res = (*CreateJavaVM)(&jvm, (void**)&env, &vmArgs); if (res < 0) { printf("error load jvm"); return -1; } } printf("after load jvm\n"); stringstream s; if (ACCESS(LOCALJARPATH, 0) != 0) { s << REMOTEJARPATH; s << "sign2fileRunnable.jar"; add_path(env, s.str().c_str()); } else { s << LOCALJARPATH; s << "sign2fileRunnable.jar"; add_path(env, s.str().c_str()); } printf("add path\n"); jclass class_Test; int file_type = -1; printf("user_name:%s\n", user_name); printf("date:%s\n", date.str().c_str()); for (auto ii = 0; ii < att_count; ii++) { if (isTypeOf(attachments[ii], "ItemRevision")) { ITKCALL(AOM_ask_value_tags(attachments[ii], "IMAN_master_form_rev", &objects_count, &objects)); //表单签名 map props; for (auto iii = 0; iii < objects_count; iii++) { ITKCALL(AOM_ask_prop_names(objects[iii], &prop_cnt, &prop_names)); AOM_lock(objects[iii]); if (strcmp("clear", sign_location) == 0) { for (int i = 0;i < pref_count;i++) { sign_date.str(""); sign_date << pref_values[i]; sign_date << "rq"; if (find(prop_cnt, prop_names, pref_values[i])) { ITKCALL(AOM_set_value_string(objects[iii], pref_values[i], "")); } else { printf("属性\"%s\"不存在", pref_values[i]); } if (find(prop_cnt, prop_names, (char*)sign_date.str().c_str())) { ITKCALL(AOM_set_value_string(objects[iii], sign_date.str().c_str(), "")); } else { printf("属性\"%s\"不存在", sign_date.str().c_str()); } } printf("sign clear\n"); } else { if (find(prop_cnt, prop_names, (char*)sign_location)) { ITKCALL(AOM_set_value_string(objects[iii], sign_location, user_name)); } else { printf("属性\"%s\"不存在", sign_location); } if (find(prop_cnt, prop_names, (char*)sign_date.str().c_str())) { ITKCALL(AOM_set_value_string(objects[iii], sign_date.str().c_str(), date.str().c_str())); } else { printf("属性\"%s\"不存在", sign_date.str().c_str()); } printf("%s,%s signoff\n", sign_location, sign_date.str().c_str()); } ITKCALL(AOM_save(objects[iii])); AOM_unlock(objects[iii]); } MEM_free(objects); printf("startSign\n"); ITKCALL(AOM_ask_value_tags(attachments[ii], "IMAN_specification", &n_attachs, &dataset)); for (int i = 0;i < n_attachs;i++) { stringstream ss; ss << dir.str().c_str(); printf(ss.str().c_str()); ITKCALL(TCTYPE_ask_object_type(dataset[i], &type_tag)); if (type_tag == wordx_tag || type_tag == word_tag) { ITKCALL(AE_ask_dataset_named_ref2(dataset[i], "word", &ae_ref, &word_ref)); char *file_name; ITKCALL(AOM_UIF_ask_value(word_ref, "file_name", &file_name)); ss << buffer; ss << "_"; ss << file_name; printf("path:%s\n", ss.str().c_str()); ITKCALL(AE_export_named_ref(dataset[i], "word", ss.str().c_str())); file_type = 0; } else if (type_tag == excel_tag || type_tag == excelx_tag) { ITKCALL(AE_ask_dataset_named_ref2(dataset[i], "excel", &ae_ref, &excel_ref)); char *file_name; ITKCALL(AOM_UIF_ask_value(excel_ref, "file_name", &file_name)); ss << buffer; ss << "_"; ss << file_name; printf("path:%s\n", ss.str().c_str()); ITKCALL(AE_export_named_ref(dataset[i], "excel", ss.str().c_str())); file_type = 1; } printf("file_type:%d\n", file_type); if (file_type == 0) { class_Test = env->FindClass("sign2file/SignToWord"); int res = sign(env, class_Test, ss.str().c_str(), user_name, (char*)sign_location, (char*)temps.str().c_str()); if (res) { replace_dataset_named_reference(dataset[i], word_ref, ss.str().c_str(), "word"); } else { return -1; } } else if (file_type == 1) { class_Test = env->FindClass("sign2file/SignToExcel"); int res = sign(env, class_Test, ss.str().c_str(), user_name, (char*)sign_location, (char*)temps.str().c_str()); if (res) { replace_dataset_named_reference(dataset[i], excel_ref, ss.str().c_str(), "excel"); } else { return -1; } } file_type = -1; } MEM_free(dataset); AOM_ask_value_tags(attachments[ii], "TC_Attaches", &n_attachs, &dataset); for (int i = 0;i < n_attachs;i++) { stringstream ss; ss << dir.str().c_str(); ITKCALL(TCTYPE_ask_object_type(dataset[i], &type_tag)); if (type_tag == wordx_tag || type_tag == word_tag) { ITKCALL(AE_ask_dataset_named_ref2(dataset[i], "word", &ae_ref, &word_ref)); char *file_name; ITKCALL(AOM_UIF_ask_value(word_ref, "file_name", &file_name)); ss << buffer; ss << "_"; ss << file_name; printf("path:%s\n", ss.str().c_str()); ITKCALL(AE_export_named_ref(dataset[i], "word", ss.str().c_str())); file_type = 0; } else if (type_tag == excel_tag || type_tag == excelx_tag) { ITKCALL(AE_ask_dataset_named_ref2(dataset[i], "excel", &ae_ref, &excel_ref)); char *file_name; ITKCALL(AOM_UIF_ask_value(excel_ref, "file_name", &file_name)); ss << buffer; ss << "_"; ss << file_name; printf("path:%s\n", ss.str().c_str()); ITKCALL(AE_export_named_ref(dataset[i], "excel", ss.str().c_str())); file_type = 1; } printf("file_type:%d\n", file_type); if (file_type == 0) { class_Test = env->FindClass("sign2file/SignToWord"); int res = sign(env, class_Test, ss.str().c_str(), user_name, (char*)sign_location, (char*)temps.str().c_str()); if (res) { replace_dataset_named_reference(dataset[i], word_ref, ss.str().c_str(), "word"); } else { return -1; } } else if (file_type == 1) { class_Test = env->FindClass("sign2file/SignToExcel"); int res = sign(env, class_Test, ss.str().c_str(), user_name, (char*)sign_location, (char*)temps.str().c_str()); if (res) { replace_dataset_named_reference(dataset[i], excel_ref, ss.str().c_str(), "excel"); } else { return -1; } } file_type = -1; } MEM_free(dataset); printf("endSign\n"); } } FreeLibrary(hModule); POM_AM__set_application_bypass(false); return 0; }