Using Python from KVS and vice-versa.
How to use Python from KVS and KVS from Python.
Introduction
Starting from version 4.0.0 you can include Python code snippets in KVS code and you can use KVS commands from within Python. This feature is present only if a working Python installation has been found at build time.
The Python support is very similar to the Perl support present since 3.x, so if you have used Perl from KVIrc before you'll find the API is almost the same.. otherwise read on :)
Using Python from KVS
Using Python from KVIrc is really easy - just enclose your Python code snippet inside python.begin and python.end.

python.begin
<python code goes here>
python.end
For example:

python.begin
f = open('myfile.txt', 'w')
f.write('This is a test\n')
f.close()
python.end
A Python code snippet can appear anywhere a KVS code snippet can with the only restriction that it must be enclosed in python.begin and python.end. This means that you can write Python code in the commandline, in the aliases, the event handlers, popups... anywhere.
If you have already encountered KVIrc's eval command then you probably also know how to execute a Python code snippet from a file :)
Using KVS from python
KVIrc exports several commands to the Python namespace that allow you to invoke KVIrc's functions from inside the Python code snippet.
The nicest example is kvirc.echo():

python.begin
kvirc.echo("Hello KVIrc world from python!")
python.end
kvirc.echo() is the counterpart of the echo. The exact syntax is:
    kvirc.echo(<text>[,<colorset>[,<windowid>]])
<text> is obviously the text to be printed. <colorset> is the equivalent of the echo -i option and <windowid> is the equivalent of the -w option. Both <colorset> and <windowid> can be omitted (in this case KVIrc will use a default colorset and the current window).
Python execution contexts
The Python code snippets are executed by a Python interpreter - each interpreter has its own context and thus its own variables, own function namespace etc.

In the example above, KVIrc creates an interpreter when python.begin is invoked and destroys it at python.end parsing time. In fact, KVIrc can maintain multiple persistent interpreters that will allow you to preserve your context across python.begin invocations.

You can invoke a specific Python context by passing it as parameter to the python.begin command:

[cmd]python.begin("mycontext")[/cmd]
myvariable = "mycontext"
kvirc.echo("This Python code is executed from " + myvariable)
python.end
The nice thing is that at a later time you can invoke this context again and discover that mycontext has preserved its value:

[cmd]python.begin("mycontext")[/cmd]
kvirc.echo("myvariable is still equal to " + myvariable)
python.end
The first time you invoke a named Python context it is automatically created and persists until KVIrc terminates or the Python context is explicitly destroyed by python.destroy.

There is a third possibility to destroy a context - when the pythoncore module is forcibly unloaded (by /pythoncore.unload). This is however a rare case and should be treated just like a KVIrc restart (the user probably WANTS the contexts to be reinitialized).

The nice thing is that not only will your variables be preserved, any python function or class you declare in a context will persist. It's just like executing a long Python script file with pauses inside.

If you omit the Python context name in the python.begin command (or if you use an empty string in its place) then KVIrc will create a temporary context for the snippet execution and will destroy it immediately after python.end has been called.

The major side effect of keeping persistent Python contexts is that python's symbol table will grow, and if not used carefully, the interpreter may become a memory hog. So if you're going to use persistent contexts, either try to keep the symbol table clean or explicitly call python.destroy once in a while to recreate the interpreter.
If you just execute occasional Python code snippets and don't need to keep persistent variables, then just use the nameless temporary context provided by python.begin("").
Passing parameters to the Python script
The easiest way to pass parameters to the Python code snippet is to put them as python.begin arguments. In fact the complete syntax of python.begin is:
python.begin(<python context>,<arg0>,<arg1>,...)
Where the <arg0>,<arg1>...<argN> parameters are passed to the Python context as elements of the aArgs array.

python.begin("","Hello world!","Now I CAN",1,2,3)
for l in range(0,5):
kvirc.echo(aArgs[l])
python.end
Accessing the KVIrc scripting context from python
KVIrc exposes the following functions that manipulate variables of KVIrc's current KVS execution context:
    kvirc.getLocal(<x>)
Returns the value of KVIrc's local variable %x.
    kvirc.getGlobal(<Y>)
Returns the value of KVIrc's global variable %Y.
    kvirc.setLocal(<x>,<value>)
Sets KVIrc's local variable %x to <value>
    kvirc.setGlobal(<Y>,<value>)
Sets KVIrc's global variable %Y to <value>
The local variables referenced belong to the current KVS execution context while the global variables are visible everywhere.

%pippo = test
%Pluto = 12345
python.begin
mypippo = kvirc.getLocal("pippo")
mypippo += " rox"
mypluto = kvirc.getGlobal("Pluto")
mypluto += " rox"
kvirc.setLocal("pippo",mypippo)
kvirc.setGlobal("Pluto",mypluto)
python.end
echo "\%pippo is" %pippo
echo "\%Pluto is" %Pluto
Executing arbitrary KVIrc commands from python
You can execute arbitrary KVS commands from Python by means of:
    kvirc.eval(<code>)
This function behaves exactly like the ${ <code> } KVS construct - it executes <code> in a child context and returns its evaluation result.
The following two code snippets have equivalent visible effects:

echo ${ return "Yeah!"; }

python.begin
kvirc.echo(kvirc.eval("return \"Yeah!\""));
python.end
You can "eval" compound command sequences and variable ones.
Remember that the Python code snippet is evaluated in a child KVS context and thus the local variables are NOT visible! The following code snippets may easily fool you:

%x = 10
python.begin
kvirc.eval("echo \"The value is %x\"")
python.end
This will print "The value is " since %x is not accessible from the eval's context. If you have tried to write something like this then you probably need to rewrite it as:

%x = 10
python.begin
x = kvirc.getLocal("x")
kvirc.eval("echo \"The value is ".$x."\"")
python.end
    
A shortcut for kvirc.eval("/say...")
    Since kvirc.eval("/say...") is a common calling pattern, say has been added     to the KVIrc Python namespace. You can now call     

kvirc.say("Hi all!")
and that will mimic the behaviour of

/say Hi all!
The complete syntax for kvirc.say() is:
    kvirc.say(<text>[,<windowid>])
and the semantics are obvious (see also /say).
Python script return values
The python.begin command propagates the Python code return value to the KVIrc context (just like a setreturn() would do) - this makes it easier to create an alias that executes a Python script and returns its result.

Without this automatic propagation, you would be forced to play with variables:
  • First use kvirc.setLocal('var', '123') from inside the python script;
  • Then, from the KVIrc script after python.end, retrieve the %var variable, check its value and call setreturn() on it.
Executing Python scripts from files

alias(pythonexec)
{
    %tmp = "python.begin(\"\",$1,$2,$3,$4,$5)";
    %tmp .= $file.read($0);
    %tmp .= "python.end";
    eval %tmp;
}
pythonexec "/home/pragma/mypythonscript.pl" "param1" "param2" "param3"
# or even
echo $pythonexec("/home/pragma/computeprimelargerthan.pl","10000")
Curiosity
The Python support in KVIrc is implemented as a master-slave module pair. The python.* module is the master while pythoncore is the slave. When Python support isn't compiled in, the python.* commands print some warnings and exit gracefully while the pythoncore module refuses to be loaded. When Python support is compiled in but for some reason the libpython.so can't be found or loaded, pythoncore fails the dynamic loading stage, however python.* only fails gracefully with warning messages. This trick allows scripters to check for python support with $python.isavailable and to embed Python code snippets in KVS even if the support is missing - the snippets will be just skipped.

Happy Python hacking :)

Index, Language Overview