/* * "Software pw3270, desenvolvido com base nos códigos fontes do WC3270 e X3270 * (Paul Mattes Paul.Mattes@usa.net), de emulação de terminal 3270 para acesso a * aplicativos mainframe. Registro no INPI sob o nome G3270. * * Copyright (C) <2008> * * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela * Free Software Foundation. * * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para * obter mais detalhes. * * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este * programa; se não, escreva para a Free Software Foundation, Inc., 59 Temple * Place, Suite 330, Boston, MA, 02111-1307, USA * * Este programa está nomeado como pluginmain.c e possui - linhas de código. * * Contatos: * * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) * * Referencias: * * * http://www.oorexx.org/docs/rexxpg/x14097.htm * * http://www.oorexx.org/docs/rexxpg/c2539.htm * */ #define ENABLE_NLS #define GETTEXT_PACKAGE PACKAGE_NAME #include "rx3270.h" #include #include #include #include #include #include #include #include #include #include #include #include #include /*--[ Globals ]--------------------------------------------------------------------------------------*/ #if GTK_CHECK_VERSION(2,32,0) static GMutex mutex; #else static GStaticMutex mutex = G_STATIC_MUTEX_INIT; #endif // GTK_CHECK_VERSION static gchar * script_name = NULL; /*--[ Rexx application data block ]--------------------------------------------------------------------------*/ struct rexx_application_data { GtkAction * action; GtkWidget * widget; GtkWidget * trace; const gchar * filename; }; /*--[ Plugin session object ]--------------------------------------------------------------------------------*/ using namespace std; using namespace PW3270_NAMESPACE; /*--[ Running rexx scripts ]---------------------------------------------------------------------------------*/ static void trace_cleanup(GtkWidget *widget, gpointer dunno) { rexx_application_data *data = (rexx_application_data *) g_object_get_data(G_OBJECT(widget),"rexx_app_data"); trace("%s: data=%p",__FUNCTION__,data); if(data) data->trace = NULL; } static GtkWidget * get_trace_window(rexx_application_data *data) { if(data->trace) return data->trace; data->trace = pw3270_trace_new(); g_signal_connect(G_OBJECT(data->trace), "destroy",G_CALLBACK(trace_cleanup), NULL); pw3270_trace_set_destroy_on_close(data->trace,TRUE); g_object_set_data(G_OBJECT(data->trace),"rexx_app_data",data); gtk_window_set_title(GTK_WINDOW(data->trace),_("Rexx trace")); gtk_window_set_transient_for(GTK_WINDOW(data->trace),GTK_WINDOW(gtk_widget_get_toplevel(data->widget))); gtk_window_set_destroy_with_parent(GTK_WINDOW(data->trace),TRUE); gtk_window_set_default_size(GTK_WINDOW(data->trace),590,430); gtk_widget_show_all(data->trace); return data->trace; } static void read_line(struct rexx_application_data *data, PRXSTRING Retstr) { gchar *value = pw3270_trace_get_command(get_trace_window(data)); if(value) { if(strlen(value) > (RXAUTOBUFLEN-1)) { Retstr->strptr = (char *) RexxAllocateMemory(strlen(value)+1); strcpy(Retstr->strptr,value); } else { g_snprintf(Retstr->strptr,RXAUTOBUFLEN-1,"%s",value); } g_free(value); } else { *Retstr->strptr = 0; } Retstr->strlength = strlen(Retstr->strptr); } static int REXXENTRY Rexx_IO_exit(RexxExitContext *context, int exitnumber, int subfunction, PEXIT parmBlock) { // trace("%s call with ExitNumber: %d Subfunction: %d",__FUNCTION__,(int) exitnumber, (int) subfunction); switch(subfunction) { case RXSIOSAY: // SAY a line to STDOUT { struct rexx_application_data *data = (struct rexx_application_data *) context->GetApplicationData(); GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(data->widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_INFO, GTK_BUTTONS_OK_CANCEL, "%s", (((RXSIOSAY_PARM *) parmBlock)->rxsio_string).strptr ); gtk_window_set_title(GTK_WINDOW(dialog), _( "Script message" ) ); if(data->trace) pw3270_trace_printf(data->trace,"%s\n",(((RXSIOSAY_PARM *) parmBlock)->rxsio_string).strptr); if(gtk_dialog_run(GTK_DIALOG (dialog)) == GTK_RESPONSE_CANCEL) context->RaiseException0(Rexx_Error_Program_interrupted); gtk_widget_destroy(dialog); } break; case RXSIOTRC: // Trace output { struct rexx_application_data *data = (struct rexx_application_data *) context->GetApplicationData(); lib3270_write_log(NULL, "rx3270", "%s", (((RXSIOTRC_PARM *) parmBlock)->rxsio_string).strptr); pw3270_trace_printf(get_trace_window(data),"%s\n",(((RXSIOTRC_PARM *) parmBlock)->rxsio_string).strptr); } break; case RXSIOTRD: // Read from char stream read_line((struct rexx_application_data *) context->GetApplicationData(), & (((RXSIODTR_PARM *) parmBlock)->rxsiodtr_retc) ); break; case RXSIODTR: // DEBUG read from char stream read_line((struct rexx_application_data *) context->GetApplicationData(), & (((RXSIODTR_PARM *) parmBlock)->rxsiodtr_retc) ); break; default: return RXEXIT_NOT_HANDLED; } return RXEXIT_HANDLED; } static void call_rexx_script(GtkAction *action, GtkWidget *widget, const gchar *filename) { const gchar * args = (const gchar *) g_object_get_data(G_OBJECT(action),"args"); struct rexx_application_data appdata; RexxInstance * instance; RexxThreadContext * threadContext; RexxOption options[25]; RexxContextExit exits[2]; memset(&appdata,0,sizeof(appdata)); appdata.action = action; appdata.widget = widget; appdata.filename = filename; memset(options,0,sizeof(options)); memset(exits,0,sizeof(exits)); exits[0].sysexit_code = RXSIO; exits[0].handler = Rexx_IO_exit; // http://www.oorexx.org/docs/rexxpg/c2539.htm options[0].optionName = DIRECT_EXITS; options[0].option = (void *) exits; options[1].optionName = APPLICATION_DATA; options[1].option = (void *) &appdata; rx3270_set_package_option(&options[2]); options[3].optionName = EXTERNAL_CALL_PATH; options[3].option = pw3270_get_datadir(NULL); trace("Rexxdir: \"%s\"",(gchar *) ((void *) options[3].option)); if(!RexxCreateInterpreter(&instance, &threadContext, options)) { GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, _( "Can't start %s script" ), "rexx" ); gtk_window_set_title(GTK_WINDOW(dialog),_( "Rexx error" )); gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),_( "Can't create %s interpreter instance" ), "rexx"); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); } else { RexxArrayObject rxArgs; trace("%s %s(%s)",__FUNCTION__,filename,args); if(args) { gchar **arg = g_strsplit(args,",",-1); size_t sz = g_strv_length(arg); rxArgs = threadContext->NewArray(sz); for(unsigned int i = 0; iArrayPut(rxArgs, threadContext->String(arg[i]), i + 1); g_strfreev(arg); } else { rxArgs = threadContext->NewArray(1); threadContext->ArrayPut(rxArgs, threadContext->String(""),1); } v3270_set_script(widget,'R',TRUE); script_name = g_path_get_basename(filename); trace("%s: Calling",filename); RexxObjectPtr result = threadContext->CallProgram(filename, rxArgs); trace("%s: Returns",filename); g_free(script_name); script_name = NULL; v3270_set_script(widget,'R',FALSE); if (threadContext->CheckCondition()) { RexxCondition condition; // retrieve the error information and get it into a decoded form RexxDirectoryObject cond = threadContext->GetConditionInfo(); threadContext->DecodeConditionInfo(cond, &condition); // display the errors GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, _( "%s script failed" ), "Rexx" ); gtk_window_set_title(GTK_WINDOW(dialog),_( "Rexx error" )); gtk_message_dialog_format_secondary_text( GTK_MESSAGE_DIALOG(dialog), _( "%s error %d: %s\n%s" ), "Rexx", (int) condition.code, threadContext->CString(condition.errortext), threadContext->CString(condition.message) ); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); } else if (result != NULLOBJECT) { CSTRING resultString = threadContext->CString(result); lib3270_write_log(NULL,"REXX","%s exits with rc=%s",filename,resultString); } instance->Terminate(); if(appdata.trace) { pw3270_trace_printf(appdata.trace,"%s","** Rexx script ends\n"); g_object_set_data(G_OBJECT(appdata.trace),"rexx_app_data",NULL); } trace("%s ends",__FUNCTION__); } g_free(options[3].option); } extern "C" { LIB3270_EXPORT void pw3270_action_rexx_activated(GtkAction *action, GtkWidget *widget) { gchar *filename = (gchar *) g_object_get_data(G_OBJECT(action),"src"); lib3270_trace_event(v3270_get_session(widget),"Action %s activated on widget %p",gtk_action_get_name(action),widget); #if GTK_CHECK_VERSION(2,32,0) if(!g_mutex_trylock(&mutex)) #else if(!g_static_mutex_trylock(&mutex)) #endif // GTK_CHECK_VERSION { GtkWidget *dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_CANCEL, "%s", _( "Can't start script" )); gtk_window_set_title(GTK_WINDOW(dialog),_( "System busy" )); gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),"%s",_( "Please, try again in a few moments" )); gtk_dialog_run(GTK_DIALOG (dialog)); gtk_widget_destroy(dialog); return; } pw3270_set_action_state(action,FALSE); if(filename && *filename) { // Has filename, call it directly call_rexx_script(action,widget,filename); } else { filename = pw3270_file_chooser(GTK_FILE_CHOOSER_ACTION_OPEN, "rexx", _( "Select script to run" ), NULL, "rex"); if(filename) { if(*filename) { call_rexx_script(action,widget,filename); } g_free(filename); } } pw3270_set_action_state(action,TRUE); #if GTK_CHECK_VERSION(2,32,0) g_mutex_unlock(&mutex); #else g_static_mutex_unlock(&mutex); #endif // GTK_CHECK_VERSION } } /*--[ Implement ]------------------------------------------------------------------------------------*/ static session * factory(const char *name) { return session::create_local(lib3270_get_default_session_handle()); } LIB3270_EXPORT int pw3270_plugin_start(GtkWidget *window, GtkWidget *terminal) { trace("%s",__FUNCTION__); #if GTK_CHECK_VERSION(2,32,0) g_mutex_init(&mutex); #endif // GTK_CHECK_VERSION rx3270_set_factory_method(factory); return 0; } LIB3270_EXPORT int pw3270_plugin_stop(GtkWidget *window, GtkWidget *terminal) { #if GTK_CHECK_VERSION(2,32,0) g_mutex_clear(&mutex); #endif // GTK_CHECK_VERSION trace("%s",__FUNCTION__); return 0; }