//============================================================================= // // File : kvi_kvs_corecallbackcommands.cpp // Created on Fri 31 Oct 2003 04:07:58 by Szymon Stefanek // // This file is part of the KVIrc IRC client distribution // Copyright (C) 2003 Szymon Stefanek // // This program is FREE software. You can redistribute it and/or // modify it under the terms of the GNU General Public License // as published by the Free Software Foundation; either version 2 // of the License, or (at your opinion) any later version. // // This program is distributed in the HOPE that it will be USEFUL, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, write to the Free Software Foundation, // Inc. ,51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. // //============================================================================= #define __KVIRC__ #include "kvi_kvs_corecallbackcommands.h" #include "kvi_kvs_kernel.h" #include "kvi_kvs_timermanager.h" #include "kvi_kvs_aliasmanager.h" #include "kvi_kvs_variantlist.h" #include "kvi_kvs_asyncdnsoperation.h" #include "kvi_kvs_eventmanager.h" #include "kvi_kvs_processmanager.h" #include "kvi_kvs_object_controller.h" #include "kvi_cmdformatter.h" #include "kvi_ircconnectionasyncwhoisdata.h" #include "kvi_ircconnection.h" #include "kvi_scriptbutton.h" #include "kvi_iconmanager.h" #include "kvi_locale.h" #include #include "kvi_tal_tooltip.h" namespace KviKvsCoreCallbackCommands { /////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: ahost @type: command @title: host @syntax: ahost [-i] [-a] ([,]){ } @short: DNS lookup @switches: !sw: --ipv6 | -i Causes the command to run in IPv6 mode !sw: --any | -a Causes the command to run in unspecified mode and lookup both IPv4 and IPv6 addresses @description: Starts a DNS lookup for the and reports the results by calling the callback routine. The -i switch causes the command to execute in IpV6 mode (and lookup ONLY IpV6 hosts!).[br] The -a switch causes the command to run in "unspecified" mode and return any available address: IpV4 or Ipv6.[br] This command also performs reverse lookups (if you pass an IP address as ).[br] The callback command gets passed five parameters:[br] $0 contains the query string ( in fact)[br] $1 contains the value 1 if the query was succesfull.[br] In that case the remaining parameters are set as follows:[br] $2 contains the first ip address associated to the [br] $3 contains the hostname associated to the [br] $4 contains the eventual passed.[br] If $1 contains the value 0 then the query has failed and[br] $2 contains an error message explaining the failure.[br] $3 is empty[br] $4 contains the eventual passed.[br] Please note that if the dns query fails to even start for some reason then your callback MAY be called even before ahost() returns.[br] @switches: !sw: -i Causes the command to execute in IpV6 mode (and lookup ONLY IpV6 hosts!). !sw: -a The -a switch causes the command to run in "unspecified" mode and return any available address: IpV4 or Ipv6. @examples: [example] ahost("localhost") { [cmd]echo[/cmd] "Lookup: "$0; if($1) { [cmd]echo[/cmd] "Ip address: "$2; [cmd]echo[/cmd] "Hostname: "$3; } else { [cmd]echo[/cmd] "Error: $2"; } } ahost -i ("irc.flashnet.it","Hello :)") { [cmd]echo[/cmd] "Lookup: "$0; [cmd]echo[/cmd] "Magic: $3"; if($1) { [cmd]echo[/cmd] "Ip address: "$2; [cmd]echo[/cmd] "Hostname: "$3; } else { [cmd]echo[/cmd] "Error: $2"; } } ahost -a ("cafe:babe::dead:beef") { [cmd]echo[/cmd] "Lookup: "$0; [cmd]echo[/cmd] "Magic: $3"; if($1) { [cmd]echo[/cmd] "Ip address: "$2; [cmd]echo[/cmd] "Hostname: "$3; } else { [cmd]echo[/cmd] "Error: $2"; } } [/example] @seealso: [cmd]host[/cmd] */ KVSCCC(ahost) { QString szQuery; KviKvsVariant * pMagicPtr; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("dnsquery",KVS_PT_NONEMPTYSTRING,0,szQuery) KVSCCC_PARAMETER("magic",KVS_PT_VARIANT,KVS_PF_OPTIONAL,pMagicPtr) KVSCCC_PARAMETERS_END KviDns::QueryType queryType = KviDns::IpV4; if(KVSCCC_pSwitches->find('i',"ipv6"))queryType = KviDns::IpV6; if(KVSCCC_pSwitches->find('a',"any"))queryType = KviDns::Any; KviKvsVariant * pMagic = pMagicPtr ? new KviKvsVariant(*pMagicPtr) : new KviKvsVariant(); KviKvsAsyncDnsOperation * op = new KviKvsAsyncDnsOperation( KVSCCC_pContext->window(), szQuery, queryType, new KviKvsScript(*KVSCCC_pCallback), pMagic); return true; } /////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: alias @title: alias @type: command @short: Adds a new alias or modifies an existing one @syntax: alias [-q] () alias [-q] (){} @switches: !sw: -q | --quiet Causes the command to run quietly @description: Adds the alias with the specified code. The implementation code can be either a single KVS instruction or an instruction block (instruction list enclosed in braces).[br] If the alias was already existing, it is replaced with the new implementation.[br] If the is empty (eg. "{}" or just a ";") the alias is removed instead of being added. If the "remove" form is used but the specified is not existing in the alias store then a warning is printed unless the -q (--quiet) switch is used. If contains a "::" prefix, then the alias is created in the namespace specified by . If the namespace is not existing, it is created. Any alias without the "::" prefix is created in the root namespace. Namespaces are useful to avoid collisions in alias names between scripts. Only really common aliases should be created in the root namespace: all your script internal functionality should be hidden in your own namespace. @examples: [example] [comment]# Add the alias j[/comment] alias(j) { [cmd]join[/cmd] $0; } [comment]# Remove the alias j[/comment] alias(j){} [comment]# Add the alias j in namespace letters[/comments] alias(letters::j) { [cmd]echo[/cmd] "j" } [/example] @seealso: [doc:kvs_aliasesandfunctions]Aliases and functions[/doc] */ /* @doc: function @title: function @type: command @short: A synomim for alias @syntax: function [-q] () function [-q] (){} @switches: !sw: -q | --quiet Causes the command to run quietly @description: This command is a synonim for [cmd]alias[/cmd]. @seealso: [doc:kvs_aliasesandfunctions]Aliases and functions[/doc] */ KVSCCC(alias) { KviKvsVariant * vName = KVSCCC_pParams->first(); if(!vName || vName->isEmpty()) { KVSCCC_pContext->error(__tr2qs("Missing alias name")); return false; } QString szName; vName->asString(szName); // we allow only [\w:]+ QRegExp re("[\\w:]+"); if(!re.exactMatch(szName)) { KVSCCC_pContext->error(__tr2qs("Alias names can contain only letters, digits, underscores and '::' namespace separators")); return false; } // make sure that we have only doubled "::" and not ":" or ":::..." QString tmp = szName; tmp.replace("::","@"); // @ is not allowed by the rule above if(tmp.find(":") != -1) { KVSCCC_pContext->error(__tr2qs("Stray ':' character in alias name: did you mean ...:: ?")); return false; } if(tmp.find("@@") != -1) { KVSCCC_pContext->error(__tr2qs("Found an empty namespace in alias name")); return false; } if(KVSCCC_pCallback->code().isEmpty()) { if(!KviKvsAliasManager::instance()->remove(szName)) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("The alias %Q is not existing"),&szName); } } else { KviKvsScript * pScript = new KviKvsScript(*KVSCCC_pCallback); pScript->setName(szName); KviKvsAliasManager::instance()->add(szName,pScript); } return true; } ///////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: awhois @type: command @title: awhois @syntax: awhois [-i] ([,]) { } @short: Asynchronous WHOIS @switches: !sw: -i | --idle-time Ask the whois informations to the server that is connected to, effectively returning the user's idle time. @description: AWHOIS stands for Asynchronous WHOIS. It is used to obtain data for a specified irc user (designated by ). This command sends a WHOIS query to the server and silently awaits the sequence of replies. When the "End of WHOIS" message is received from server the is executed passing the WHOIS information as positional parameters.[br] The string is an optional string to be evaluated at AWHOIS execution time. It is passed as the last positional parameter.[br] Callback command parameters:[br] $0 = nickname[br] $1 = username[br] $2 = hostname[br] $3 = realname (may be empty)[br] $4 = server[br] $5 = idle time (may be empty)[br] $6 = signon time (may be empty)[br] $7 = channels (may be empty)[br] $8 = server that provided the information[br] $9 = special information (may be empty)[br] $10 = magic string evaluated at awhois call (may be empty)[br] If the -i switch is specified , the whois message is sent to the server that the user is connected to; in this way you will probably get the idle time of the user too.[br] If the server replies with a "No such nick/channel error message" the will be still triggered , but will have all the parameters empty with the exception of $0.[br] If the connection gets interrupted before all the information have been received, the will never be triggered.[br] This command is [doc:connection_dependant_commands]connection dependant[/doc].[br] @examples: [example] awhois(pragma){ echo $0-; } [/example] */ KVSCCC(awhois) { QString szNick; KviKvsVariant * pMagic; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("nickname",KVS_PT_NONEMPTYSTRING,0,szNick) KVSCCC_PARAMETER("magic",KVS_PT_VARIANT,KVS_PF_OPTIONAL,pMagic) KVSCCC_PARAMETERS_END KVSCCC_REQUIRE_CONNECTION KviQCString szN = KVSCCC_pConnection->encodeText(szNick); KviAsyncWhoisInfo * info = new KviAsyncWhoisInfo(); info->pCallback = new KviKvsScript(*KVSCCC_pCallback); info->pMagic = pMagic ? new KviKvsVariant(*pMagic) : new KviKvsVariant(); info->szNick = szNick; info->pWindow = KVSCCC_pWindow; KVSCCC_pConnection->asyncWhoisData()->add(info); if(KVSCCC_pSwitches->find('i',"idle-time"))KVSCCC_pConnection->sendFmtData("WHOIS %s %s",szN.data(),szN.data()); else KVSCCC_pConnection->sendFmtData("WHOIS %s",szN.data()); return true; } ///////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: button @title: button @type: command @short: Adds/removes/changes user definable buttons @syntax: button [-d] [-q] (,[,[,]]) { } @switches: !sw: -d | --disabled Creates the button as disabled !sw: -q | --quiet Run quietly, print no warnings @description: Adds a new user defined button with the specified .[br] [br] The parameter is ignored and is present only for backward compatibility. [br] The button image is specified by the [doc:image_id][/doc].[br] The optional button text is specified by .[br] The will be executed as reaction to a button press.[br] [br] The "window" type button can be added only to the windows that have a button container: this actually includes at least console , channels and queries.[br] The button is added to the current window; if you want to add it to a different window , use the [doc:command_rebinding]standard -r command rebinding[/doc] switch.[br] The will be executed as reaction to a button press; the code execution will be bound to the window that the button is attacched to.[br] If a button with already exists in the current window, its parameters are changed according to the passed values (, and ).[br] [br] Passing an empty removes the button.[br] The callback parameters $0 and $1 will contain the screen coordinates of the bottom-left corner of the button: this is useful for showing a popup menu in response to the click.[br] If the -q switch is used , this command prints no warnings.[br] The -d switch causes the button to be disabled (grayed).[br] @examples: [example] button(w,test,-1,Test button){ echo Test!; } button(w,test){} [/example] */ KVSCCC(button) { KviKvsVariant * pUnused; QString szName,szIcon,szLabel; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("type_unused",KVS_PT_VARIANT,0,pUnused) KVSCCC_PARAMETER("name",KVS_PT_NONEMPTYSTRING,0,szName) KVSCCC_PARAMETER("icon",KVS_PT_NONEMPTYSTRING,KVS_PF_OPTIONAL,szIcon) KVSCCC_PARAMETER("label",KVS_PT_NONEMPTYSTRING,KVS_PF_OPTIONAL,szLabel) KVSCCC_PARAMETERS_END KviScriptUserButton * pButton = 0; if(!KVSCCC_pWindow->buttonContainer()) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("The specified window has no button containers")); return true; } pButton = (KviScriptUserButton *)(KVSCCC_pWindow->buttonContainer())->child(szName,"KviWindowScriptButton"); if(KVSCCC_pCallback->code().isEmpty()) { if(pButton)delete pButton; else { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("Window button '%Q' not found"),&szName); } return true; } if(!pButton) { pButton = new KviWindowScriptButton(KVSCCC_pWindow->buttonContainer(),KVSCCC_pWindow,szName); pButton->show(); } KviTalToolTip::remove(pButton); if(!szLabel.isEmpty()) { pButton->setButtonText(szLabel); KviTalToolTip::add(pButton,szLabel); } pButton->setButtonCode(new KviKvsScript(*KVSCCC_pCallback)); if(!szIcon.isEmpty()) { QPixmap * pix = g_pIconManager->getImage(szIcon); if(pix) { pButton->setButtonPixmap(*pix); } else { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("Can't find the icon '%Q'"),&szIcon); } } pButton->setEnabled(!(KVSCCC_pSwitches->find('d',"disabled"))); return true; } ///////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: event @title: event @type: command @short: Adds a new event handler @syntax: event [-q] (,) { } @switches: !sw: -q | --quiet Do not print any warnings @description: Adds the handler with to the list of handlers for the event .[br] If the is empty the handler is removed from the handler list instead of being added.[br] The may be one of the kvirc-builtin event names or a numeric code (from 0 to 999) of a raw server message.[br] If the -q switch is specified then the command runs in quiet mode. @seealso: [cmd]eventctl[/cmd] */ KVSCCC(event) { QString szEventName,szHandlerName; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("event_name",KVS_PT_NONEMPTYSTRING,0,szEventName) KVSCCC_PARAMETER("handler_name",KVS_PT_NONEMPTYSTRING,0,szHandlerName) KVSCCC_PARAMETERS_END bool bOk; int iNumber = szEventName.toInt(&bOk); bool bIsRaw = (bOk && (iNumber >= 0) && (iNumber < 1000)); if(bIsRaw) { if(!KviKvsEventManager::instance()->isValidRawEvent(iNumber)) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("No such event (%Q)"),&szEventName); return true; } } else { iNumber = KviKvsEventManager::instance()->findAppEventIndexByName(szEventName); if(!KviKvsEventManager::instance()->isValidAppEvent(iNumber)) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("No such event (%Q)"),&szEventName); return true; } } if(KVSCCC_pCallback->code().isEmpty()) { if(bIsRaw) { if(!KviKvsEventManager::instance()->removeScriptRawHandler(iNumber,szHandlerName)) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("No handler '%Q' for raw numeric event '%d'"),&szHandlerName,iNumber); } } else { if(!KviKvsEventManager::instance()->removeScriptAppHandler(iNumber,szHandlerName)) { if(!KVSCCC_pSwitches->find('q',"quiet")) KVSCCC_pContext->warning(__tr2qs("No handler '%Q' for event '%Q'"),&szHandlerName,&szEventName); } } } else { if(bIsRaw) { // remove the old handler KviKvsEventManager::instance()->removeScriptRawHandler(iNumber,szHandlerName); QString contextName; KviQString::sprintf(contextName,"RawEvent%d::%Q",iNumber,&szHandlerName); KviKvsScriptEventHandler * pHandler = new KviKvsScriptEventHandler(szHandlerName,contextName,KVSCCC_pCallback->code()); KviKvsEventManager::instance()->addRawHandler(iNumber,pHandler); } else { // remove the old handler KviKvsEventManager::instance()->removeScriptAppHandler(iNumber,szHandlerName); QString contextName; KviQString::sprintf(contextName,"%Q::%Q",&szEventName,&szHandlerName); KviKvsScriptEventHandler * pHandler = new KviKvsScriptEventHandler(szHandlerName,contextName,KVSCCC_pCallback->code()); KviKvsEventManager::instance()->addAppHandler(iNumber,pHandler); } } return true; } /////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: exec @type: command @title: exec @syntax: exec [switches] ([,]) { } @short: Asynchronous execution of external programs @switches: !sw: -q | --quiet Quiet: do not print any warnings !sw: -t | --trigger-termination Trigger the termination event !sw: -x | --trigger-startup Trigger the startup event !sw: -n | --no-stdout Do NOT trigger any stdout events !sw: -e | --trigger-stderr Trigger stderr events !sw: -b | --output-block Trigger the with the stdout and stderr events exactly once, passing the complete block of process output. The events are triggered even if the process output is empty. !sw: -k= | --kill-after= Kill the process unconditionally after milliseconds. If the -t switch is used then the termination event will be triggered just after the process has been killed. !sw: -p= | --trigger-ping= Trigger with "ping" events every milliseconds. !sw: -w | --bind-to-window Kill the process if the current window is closed. In this case the termination event is NOT triggered (since the parent window has been lost). If this switch is not used then the process is rebound to the active console window and continues running. !sw: -s= | --shell= Use instead of the default interpreter "sh -c". The should be able to launch the interpeter and should contain the necessary arguments in order to allow KVirc to pass the "commandline" by appending it as the last parameter. !sw: -d | --direct Use no command interpreter at all: run the command directly. Takes precedence over -s. !sw: -q | --quiet Run quietly @description: [b]Overview[/b][br] Executes the by passing it to a command interpreter. The is executed asynchronously: this means that when exec returns the control to the next command, may be still running.[br] [br] [b]The callback[/b][br] The is triggered on several events related to the child process and it gets passed the following parameters:[br] $0 = [br] $1 = [br] $2 = [br] The first parameter specifies the event cause and contains one of the following strings: "stdout","stderr","terminated","started" and "ping". [b]By default (if no switches are used) only "stdout" type events are triggered[/b]. The second parameter depends on the event cause and contains data sensible to each event type. The third parameter is the eventual passed to the exec command call.[br] [br] [b]Interacting with the process[/b][br] If you use [cmd]halt[/cmd] to terminate the callback then the slave process is killed immediately and no other callback events are triggered.[br] If you return some non empty string then this string will be written to the process stdin stream. This trick can be used to control interactive processes. Please note that you must include all the relevant carriage returns and newlines in the return value (see [fnc]$cr[/fnc] and [fnc]$lf[/fnc]).[br] [br] [b]Startup event[/b][br] If the -x switch is used then the startup event is triggered just after the process has been succesfully launched. The $0 parameter passed to the callback contains the string "started". Parameter $1 contains the pid of the slave process.[br] [br] [b]Stdout data event[/b][br] The stdout data event is triggered when the process prints some output on its stdout stream. This event is triggered by default and to disable it you must use the -n switch. $0 contains the string "stdout". If the -b switch is not used then $1 contains a single line of process output with the trailing carriage return and/or line feed stripped. If -b is used then $1 contains the whole process output block (eventually empty) with all the cr/lf pairs.[br] [br] [b]Stderr data event[/b][br] The stderr data event is similar to the stdout one but there are three differences. The first one is that the stderr event is NOT triggered by default: you must use the -e switch to enable it. The second difference is that $0 contains "stderr" instead of "stdout". The last difference is that $1 contains data coming from the slave process stderr stream.[br] [br] [b]Termination event[/b][br] The termination event is triggered after the slave process has terminated its execution. You must use the -t switch to enable it since it is disabled by default. $0 contains the string "terminated". $1 contains the process exit status value. (Note that if the process has crashed or has been terminated by an external singnal then this value will be 0).[br] [br] [b]Ping event[/b][br] The ping event is triggered only if the -p= switch is passed.[br] This event may be useful to monitor the process status while it is not emitting any output, to write data to its stdin stream (by the means of [cmd]return[/cmd]) or simply to give some feedback to the user while the slave process is doing a long computation.[br] [br] [b]The extended scope variables[/b][br] The has a set of [doc:data_types]extended scope variables[/doc] that conserve their value during the whole life time of the slave process.[br] These variables can be accessed through the %: syntax and are useful to store process private data between multiple calls.[br] [b]Some words about the switches[/b][br] If the -b switch is used then the is called only once for the events stdout and stderr (if enabled) with the complete output block from the process. With the -b switch the events stdout and stderr are triggered once even if the process emits no output. The -s= switch may be used to specify the path of the command interpreter that is "sh -c" by default on unix machines and "cmd.exe /c" on windows. The interpreter executable is searched on the system PATH. If the process can't be started then a warning message is printed in the current window unless the -q (quiet) flag is used.[br] [br] @examples: [example] [comment]# Really simple example: print only the stdout of a slave process[/comment] exec("cat /proc/cpuinfo"){ echo $1; }; [comment]# Now print only stderr: enable stderr and disable stdout[/comment] exec -e -n ("sed -senseless"){ echo $1; }; [comment]# Do it another way: enable stderr and filter out stdout[/comment] exec -e ("sed -senseless"){ if($0 == "stderr")echo $1; } [comment]# Now enable all (almost) events and print them[/comment] exec -e -t -s ("cat /proc/cpuinfo && sed -senseless"){ echo [event:$0] $1; } [comment]# Now see what happens if -b is used[/comment] exec -b -e -t -s ("cat /proc/cpuinfo && sed -senseless"){ echo [event:$0] $1; } [comment]# Run an iterative script and kill it after 20 seconds[/comment] exec -k=20000 ("while true; do sleep 1; echo \"Tic\"; done"){ echo [event:$0] $1; } [comment]# Run a blocking process, kill it after 20 seconds[/comment] [comment]# and give feedback to the user by the means of ping[/comment] exec -k=20000 -p=1000 -t ("cat") { if($0 == "ping")echo "[event:$0] Please wait while doing a huge computation ..." else if($0 == "terminated")echo "[event:$0] Ok, done :)" } [comment]# Do the same but this time use the extended scope vars[/comment] [comment]# Use also a nicer syntax[/comment] exec -k=20000 -p=1000 -t ("cat") { switch($0) { case("ping"): { if(%:x == 1) { %:x = 0; echo "Tic!" } else { %:x = 1; echo "Tac!" } } break; case("terminated"): { echo "Ok, done :)" } break; } } [comment]# Again do the same but kill the process explicitly[/comment] exec -x -p=1000 -t ("cat") { switch($0) { case("started"): { [comment]# Initialize the counter[/comment] %:x = 10; } break; case("ping"): { echo %:x %:x-- [comment]# When the counter reaches zero, kill the process with halt[/comment] if(%:x == 0)halt; } break; case("terminated"): { echo "Boom!" } break; } } [comment]# Now play with an interactive process[/comment] [comment]# WARNING: Please note that spam is illegal and generates bad karma[/comment] [comment]# Try it only with your own e-mail address as recipient[/comment] exec -s -k=60000 -t ("telnet my.mail.server.com 25") { if($0 == "started") { %:state = 0 [comment]# Returning an empty string does not write to stdin[/comment] return } if($1 == "stderr") { echo "[stderr] $1" return } if($1 == "terminated") { echo "[process terminated]" return } echo "[stdout] $1" switch(%:state) { case(0): { [comment]# Waiting for 220 (ready)[/comment] if($str.match("220*",$1)) { %:state++ echo "Sending HELO..." return "HELO myhostname$cr$lf"; } } break case(1): { [comment]# Waiting for 250 (after the HELO)[/comment] if($str.match("250*",$1)) { %:state++ echo "Sending MAIL..." return "MAIL From: $cr$lf" } else { echo "HELO command not accepted: $1" halt } } break; case(2): { [comment]# Waiting for another 250 (MAIL accepted)[/comment] if($str.match("250*",$1)) { %:state++ echo "Sending RCPT..." return "RCPT To: $cr$lf" } else { echo "MAIL command not accepted: $1" halt } } break; case(3): { [comment]# Waiting for another 250 (RCPT accepted)[/comment] if($str.match("250*",$1)) { %:state++ echo "Sending DATA..." return "DATA$cr$lf" } else { echo "RCPT not accepted: $1" halt } } break; case(4): { [comment]# Waiting for 354 (ok, go on)[/comment] if($str.match("354*",$1)) { %:state++ echo "Sending body..." return "This is a test message :)$cr$lf$cr$lf.$cr$lf" } else { echo "Mail body not accepted: $1" halt } } break; case(5): { [comment]# We don't wait anymore :)[/comment] %:state++ echo "Sending QUIT..." return "QUIT$cr$lf" } break; default: { [comment]# Usually the mail server closes the connection[/comment] %:state++ if(%:state > 10) { [comment]# But if it does not in few messages[/comment] [comment]# Then force the process to die[/comment] halt } } } } [/example] */ KVSCCC(exec) { QString szCommandline; KviKvsVariant * pMagic; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("commandline",KVS_PT_NONEMPTYSTRING,0,szCommandline) KVSCCC_PARAMETER("magic",KVS_PT_VARIANT,KVS_PF_OPTIONAL,pMagic) KVSCCC_PARAMETERS_END int f = 0; if(KVSCCC_pSwitches->find('t',"trigger-termination") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_TRIGGERTERMINATED; if(KVSCCC_pSwitches->find('n',"no-stdout") == 0)f |= KVI_KVS_PROCESSDESCRIPTOR_TRIGGERSTDOUT; if(KVSCCC_pSwitches->find('e',"trigger-stderr") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_TRIGGERSTDERR; if(KVSCCC_pSwitches->find('x',"trigger-startup") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_TRIGGERSTARTED; if(KVSCCC_pSwitches->find('b',"output-block") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_OUTPUTBYBLOCKS; if(KVSCCC_pSwitches->find('w',"bind-to-window") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_KILLIFNOWINDOW; if(KVSCCC_pSwitches->find('d',"direct") != 0)f |= KVI_KVS_PROCESSDESCRIPTOR_NOSHELL; QString szShell; KVSCCC_pSwitches->getAsStringIfExisting('s',"shell",szShell); kvs_int_t iPingTime = 0; kvs_int_t iMaxRunTime = 0; KviKvsVariant * pPing = KVSCCC_pSwitches->find('p',"trigger-ping"); if(pPing) { if(!(pPing->asInteger(iPingTime) && iPingTime > 0)) { KVSCCC_pContext->warning(__tr2qs("The specified ping time is invalid: assuming zero (no ping)")); iPingTime = 0; } } KviKvsVariant * pKill = KVSCCC_pSwitches->find('k',"kill-after"); if(pKill) { if(!(pKill->asInteger(iMaxRunTime) && iMaxRunTime > 0)) { KVSCCC_pContext->warning(__tr2qs("The specified maximum run time is invalid: assuming zero (infinite)")); iMaxRunTime = 0; } } KviKvsProcessDescriptorData * d = new KviKvsProcessDescriptorData; d->szCommandline = szCommandline; d->szShell = szShell; d->pWnd = KVSCCC_pContext->window(); d->pMagic =pMagic ? new KviKvsVariant(*pMagic) : 0; d->iFlags = f; d->pCallback = new KviKvsScript(*KVSCCC_pCallback); d->iMaxRunTime = iMaxRunTime; d->iPingTimeout = iPingTime; KviKvsProcessAsyncOperation * op = new KviKvsProcessAsyncOperation(d); if(!op->start()) { if(KVSCCC_pSwitches->find('q',"quiet") == 0)KVSCCC_pContext->warning(__tr2qs("Failed to start the process")); delete op; } return true; } ////////////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: privateimpl @title: privateimpl @type: command @short: Adds a private implementation of a function @syntax: privateimpl(,) @description: Adds a private implementation of function to the existing object designed by . must be a valid command sequence.[br] Side note:[br] This command can not succesfully implement the "constructor" function since it must be called after this one has already been executed.[br] To implement a constructor you MUST write your own class definition.[br] @seealso: [cmd]class[/cmd], [doc:objects]Objects documentation[/doc] */ KVSCCC(privateimpl) { kvs_hobject_t hObject; QString szFunctionName; KVSCCC_PARAMETERS_BEGIN KVSCCC_PARAMETER("object_handle",KVS_PT_HOBJECT,0,hObject) KVSCCC_PARAMETER("function_name",KVS_PT_NONEMPTYSTRING,0,szFunctionName) KVSCCC_PARAMETERS_END KviKvsObject * o = KviKvsKernel::instance()->objectController()->lookupObject(hObject); if(!o) { KVSCCC_pContext->error(__tr2qs("The specified object does not exist")); return false; } o->registerPrivateImplementation(szFunctionName,KVSCCC_pCallback->code()); return true; } ///////////////////////////////////////////////////////////////////////////////////////////////////////// /* @doc: timer @title: timer @type: command @short: Starts a timer @syntax: timer [-s] [-p] (,[,[,[,...]]]) { } @switches: !sw: -s | --single-shot Causes the timer to trigger only once (single shot timer) !sw: -p | --persistent Creates a persistent timer bound to any existing window @description: Starts a new timer named with the specified delay (in milliseconds).[br] The timer periodically calls the specified code passing the eventual strings as positional parameters.[br] If a timer with the same name already exists, it is replaced by this one.[br] [b]The is evaluated at timer "shot" time and NOT while this command is being parsed. This means that the identifiers that you put inside will NOT have the current values.[/b] The values will be assigned at timer "shot" time.[br] This is a common scripters error and problem: if it is not clear, look at the examples below.[br] The timer is bound to the window in that this command is executed in.[br] If the window gets destroyed, the timer is stopped; unless the -p switch is used.[br] The -p switch causes the timer to be persistent across the application and exists until the last window has been closed: it is basically rebound to another (random) window when the original window is destroyed.[br] The -s switch cuases this timer to trigger only once: it will be automatically destroyed after that.[br] The time has an associated set of [doc:data_structures]extended scope variables[/doc]: the variables that begin with "%:" have their life extended to the whole "life" of the timer.[br] Using a very low delay is a common method to perform some background processing: you basically split a huge job in small slices and execute them when the timer is triggered until you run out of slices. A delay of 0 will cause the timer to be called whenever KVIrc has some "idle time" to spend. On the other hand, remember that timers are precious resources: many timers running with a very low delay will cause KVIrc to slow down.[br] Since all the kvirc timers share the same namespace it is a good idea to use descriptive timer names: a timer named "a" is likely to be used by two or more scripts at once causing one (or both) of them to fail.[br] A timer can be stopped at any time by using the [cmd]killtimer[/cmd] command. @seealso: [cmd]killtimer[/cmd] @examples: [example] [comment]# Just a plain timer[/comment] timer(test,1000){ echo "Hello!"; } [comment]# Now watch the timer running[/comment] killtimer test [comment]# Single shot timer[/comment] timer -s (test,1000){ echo "This will fire only once!"; } [comment]# The call above is equivalent to[/comment] timer(test,1000){ echo "This will file only once!"; killtimer test; } [comment]# Callback parameters: consider the following code[/comment] %parameter = "some string value" echo "Before calling /timer \%parameter is \"%parameter\"" timer -s (test,1000,%parameter){ echo "inside the callback \%parameter is \"%parameter\" but \$0 is \"$0\""; } [comment]# watch the timer running , and note the behaviour of the %parameter variable[/comment] killtimer test [comment]# Use the extended scope timer variables[/comment] timer(test,1000) { [comment]# Use the extended scope %:count variable to keep track[/comment] [comment]# of the times that this timer has been called[/comment] [cmd]if[/cmd]("%:count" == "")%:count = 1 else %:count++ [cmd]echo[/cmd] "This timer has fired %:count times" if(%:count == 10) { # This will kill the current timer, we don't need to specify the name [cmd]killtimer[/cmd] } } [comment]# Use isTimer to check if the timer exists[/comment] [cmd]echo[/cmd] [fnc]$isTimer[/fnc](test) [comment]# Repeat the command above after the 10th timeout...[/comment] [/example] */ KVSCCC(timer) { KviKvsVariant * vName = KVSCCC_pParams->first(); KviKvsVariant * vDelay = KVSCCC_pParams->next(); if(!vName || vName->isEmpty()) { KVSCCC_pContext->error(__tr2qs("Missing timer name")); return false; } QString szName; vName->asString(szName); if(!vDelay) { KVSCCC_pContext->error(__tr2qs("Missing timeout delay")); return false; } kvs_int_t iDelay; if(!vDelay->asInteger(iDelay)) { KVSCCC_pContext->error(__tr2qs("The timeout delay didn't evaluate to an integer")); return false; } KviKvsTimer::Lifetime lt; if(KVSCCC_pSwitches->find('s',"single-shot"))lt = KviKvsTimer::SingleShot; else if(KVSCCC_pSwitches->find('p',"persistent"))lt = KviKvsTimer::Persistent; else lt = KviKvsTimer::WindowLifetime; // prepare the callback parameters KviKvsVariantList * l = new KviKvsVariantList(); l->setAutoDelete(true); KviKvsVariant * v = KVSCCC_pParams->next(); while(v) { l->append(new KviKvsVariant(*v)); // copy v = KVSCCC_pParams->next(); } if(!KviKvsTimerManager::instance()->addTimer(szName,lt,KVSCCC_pContext->window(),iDelay,new KviKvsScript(*KVSCCC_pCallback),l)) { KVSCCC_pContext->error(__tr2qs("Unable to add the timer: insufficient system resources")); return false; } return true; } void init() { KviKvsKernel * pKern = KviKvsKernel::instance(); #define _REGCMD(__cmdName,__routine) \ { \ KviKvsCoreCallbackCommandExecRoutine * r = new KviKvsCoreCallbackCommandExecRoutine; \ r->proc = KVI_PTR2MEMBER(KviKvsCoreCallbackCommands::__routine); \ pKern->registerCoreCallbackCommandExecRoutine(QString(__cmdName),r); \ } #ifdef COMPILE_NEW_KVS _REGCMD("ahost",ahost); _REGCMD("awhois",awhois); #endif _REGCMD("alias",alias); #ifdef COMPILE_NEW_KVS _REGCMD("button",button); _REGCMD("event",event); _REGCMD("exec",exec); _REGCMD("privateimpl",privateimpl); #endif _REGCMD("function",alias); _REGCMD("timer",timer); #undef _REGCMD } };