Wikipedia:Guide to Scribbling



This is the Guide to Scribbling. Scribbling, also known as Luafication, is the act of writing a template, or converting a template, so that it uses the Scribunto extension to MediaWiki. The Scribunto extension was developed by Tim Starling and Victor Vasiliev, and allows for embedding scripting languages in MediaWiki. Currently the only supported scripting language is Lua. This Guide aims to give you a broad overview of Scribbling, and pointers to further information in various places.

Scribbled templates come in two parts: the template itself and one or more back-end modules &mdash; in the  namespace &mdash; that contain programs that are run on the wiki servers to generate the wikitext that the template expands to. The template invokes a function within a module using a new parser function named.

The idea of Scribbling is to improve template processing performance. Scribbling eliminates any need for template parser function programming using parser functions such as, , and. All of this is instead done in the module, in a language that was actually designed to be a programming language, rather than a template system onto which was bolted various extensions over time to try to make it into a programming language. Scribbling also eliminates any need for templates to expand to further templates and potentially hit the expansion depth limit. A fully Scribbled template should never need to transclude other templates.

Lua
The language in which modules are written is Lua. Unlike the template parser function system, Lua was actually designed not only to be a proper programming language, but also to be a programming language that is suitable for what is known as embedded scripting. Modules in MediaWiki are an example of embedded scripts. There are several embedded scripting languages that could have been used, including REXX and tcl; and indeed the original aim of Scribunto was to make available a choice of such languages. At the moment, however, only Lua is available.

The official reference manual for Lua is. It's a reference, not a tutorial. Consult it if you want to know the syntax or semantics for something. For a tutorial, see either  ( is also available, although it is of course out of date.) or. The downsides to these books are that quite a lot of the things that they tell you about have no bearing upon using Lua in MediaWiki modules. You don't need to know how to install Lua and how to integrate its interpreter into a program or run it standalone. The MediaWiki developers have done all of that. Similarly, a lot of the Lua library functions are, for security, not available in modules. (For example, it's not possible to do file I/O or to make operating system calls in MediaWiki modules.) So, much of what these books explain about Lua standard library functions and variables that come with the language is either irrelevant or untrue here.

The original API specification &mdash; the Lua standard library functions and variables that are supposed to be available in modules &mdash; is given at MW:Extension:Scribunto/API specification. However, even that is untrue. What you'll actually have available is documented in MW:Extension:Scribunto/Lua reference manual, which is a cut down version of the 1st Edition Lua manual that has been edited down and modified by Tim Starling to bring it more into line with the reality of Scribbling. Again, though, this is a reference manual, not a tutorial.

The things in Lua that you will mostly be concerned with, writing Scribbled templates, are tables, strings numbers, booleans,,  ,  ,   (generated  ),   (numerical  ),  ,  ,  ,  ,  , expressions and the various operators (including  ,  , the arithmetic operators  ,  ,  ,  ,  , and  ), and the  ,  , and   global tables (i.e. libraries).

Template structure
This is simple. Your template comprises one expansion of  in the usual case. Here is Harvard citation, for example:

If you find yourself wanting to use other templates within your template, or to use template parser functions, or indeed anything at all other than  and possibly some variables as its arguments, then you are using the wrong approach.

Overall structure
Let's consider a hypothetical module, Module:Population. (See Module:Population clocks for a similar, but more complex, module.) It can be structured in one of two ways:

Execution
The execution of a module by  is actually twofold:
 * 1) The module is loaded and the entire script is run.  This loads up any additional modules that the module needs (using the   function), builds the (invocable) functions that the module will provide to templates, and returns a table of them.
 * 2) The function named in   is picked out of the table built in phase 1 and called, with the arguments supplied to the template and the arguments supplied to   (more on which later).

The first Lua script does phase 1 fairly explicitly. It creates a local variable named  on line 1, initialized to a table; builds and adds a function to it (lines 3–5), by giving the function the name   in the table named by   (  being the same as saying  ); and then returns (on line 7) the table as the last line of the script. To expand such a script with more (invocable) functions, one adds them between the  statement at the top and the   statement at the bottom. (Non-invocable local functions can be added before the  statement.)  The local variable doesn't have to be named. It could be named any valid Lua variable name that you like. is simply conventional for this purpose, and is also the name that you can use to test the script in the debug console of the Module editor.

The second Lua script does the same thing, but more "idiomatically". Instead of creating a named variable as a table, it creates an anonymous table on the fly, in the middle of the  statement, which is the only (executed during the first phase) statement in the script. The  on lines 2–4 creates an (also anonymous) function and inserts it into the table under the name. To expand such a script with more (invocable) functions, one adds them as further fields in the table. (Non-invocable local functions can, again, be added before the  statement.)

In both cases, the template code that one writes is  to invoke the function named   from the module Module:Population. Also note that  builds a function, as an object, to be called. It doesn't declare it, as you might be used to from other programming languages, and the function isn't executed until it is called.

One can do more complex things than this, of course. For example: One can declare other local variables in addition to , to hold tables of data (such as lists of Language or country names), that the module uses. But this is the basic structure of a module. You make a table full of stuff, and return it.

Receiving template arguments
An ordinary function in Lua can take an (effectively) arbitrary number of arguments. Witness this function from Module:Wikitext that can be called with anywhere between zero and three arguments:

Functions called by  are special. They expect to be passed exactly one argument, a table that is called a frame (and so is conventionally given the parameter name  in the parameter list of the function). It's called a frame because, unfortunately, the developers chose to name it for their convenience. It's named after an internal structure within the code of MediaWiki itself, which it sort of represents.

This frame has a (sub-)table within it, named. It also has a means for accessing its parent frame (again, named after a thing in MediaWiki). The parent frame also has a (sub-)table within it, also named.
 * The arguments in the (child, one supposes) frame &mdash; i.e. the value of the  parameter to the function &mdash; are the arguments passed to   within the wikitext of your template.  So, for example, if you were to write   in your template then the arguments sub-table of the child frame would be (as written in Lua form).
 * The arguments in the parent frame are the arguments passed to your template when it was transcluded. So, for example, were the user of your template to write   then the arguments sub-table of the parent frame would be (as written in Lua form).

A handy programmers' idiom that you can use, to make this all a bit easier, is to have local variables named (say)  and   in your function, that point to these two argument tables. See this, from Module:WikidataCheck:

Everything in  is thus an argument that you have specified, in your template, that you can reference with code such as   and. These will be things that tell your module function its "configuration" (e.g. a CSS class name that can vary according to what template is used).

Everything in  is thus an argument that the user of the template has specified, where it was transcluded, that you can reference with code such as   and. These will be the normal template arguments, as documented on your template's  page.

See other places and other ships for two templates that both do  but do so with different arguments in place of the , thereby obtaining different results from one single common Lua function.

For both sets of arguments, the name and value of the argument are exactly as in the wikitext, except that leading and trailing whitespace in named parameters is discounted. This has an effect on your code if you decide to support or employ transclusion/invocation argument names that aren't valid Lua variable names. You cannot use the "dot" form of table lookup in such cases. For instance:  is, as you can see from the syntax colourization here, not a reference to an   argument, but a reference to an   argument and a   variable with the subtraction operator in the middle. To access such an argument, use the "square bracket" form of table lookup:.

Named arguments are indexed in the  table by their name strings, of course. Positional arguments (whether as the result of an explicit  or otherwise) are indexed in the   tables by number, not by string. is not the same as, and the latter is effectively unsettable from wikitext.

Finally, note that Lua modules can differentiate between arguments that have been used in the wikitext and simply set to an empty string, and arguments that aren't in the wikitext at all. The latter don't exist in the  table, and any attempt to index them will evaluate to. Whereas the former do exist in the table and evaluate to an empty string,.

Errors
Let's get one thing out of the way right at the start: '' Script error is a hyperlink. You can put the mouse pointer on it and click.''

We've become so conditioned by our (non-Scribbled) templates putting out error messages in red that we think that the Scribunto "Script error" error message is nothing but more of the same. It isn't. If you have JavaScript enabled in your WWW browser, it will pop up a window giving the details of the error, a call backtrace, and even hyperlinks that will take you to the location of the code where the error happened in the relevant module.

You can cause an error to happen by calling the  function.

Arguments tables are "special".
For reasons that are out of the scope of this Guide, the  sub-table of a frame is not quite like an ordinary table. It starts out empty, and it is populated with arguments as and when you execute code that looks for them. (It's possible to make tables that work like this in a Lua program, using things called metatables. That, too, is outwith the scope of this Guide.)

An unfortunate side-effect of this is that some of the normal Lua table operators don't work on an  table. The length operator,, will not work, and neither will the functions in Lua's   library. These only work with standard tables, and fail when presented with the special  table. However, the  and   functions will both work, as code to make their use possible has been added by the developers.

Copy table contents into local variables.
A name in Lua is either an access of a local variable or a table lookup. is a table lookup (of the string ) in the (global)   table, for example. Table lookups are slower, at runtime, than local variable lookups. Table lookups in tables such as the  table with its "specialness" are a lot slower.

A function in Lua can have up to 250 local variables. So make liberal use of them:
 * If you call  many times, copy it into a local variable and use that instead:
 * Don't use  over and over.  Copy it into a local variable and use that: (Even the   variable itself is a way to avoid looking up   in the   table over and over.)

When copying arguments into local variables there are two useful things that you can do along the way:
 * The alternative names for the same argument trick. If a template argument can go by different names &mdash; such as uppercase and lowercase forms, or different English spellings &mdash; then you can use Lua's   operator to pick the highest priority name that is actually supplied: This works for two reasons:
 * is the same as  as far as   is concerned.
 * Lua's  operator has what are known as "shortcut" semantics.  If the left-hand operand evaluates to something that isn't   or , it doesn't bother even working out the value of the right-hand operand.  (So whilst that first example may at first glance look like it does four lookups, in the commonest case, where   is used with the template, it in fact only actually does one.)
 * The default to empty string trick. Sometimes the fact that an omitted template argument is   is useful.  Other times, however, it isn't, and you want the behaviour of missing arguments being empty strings.  A simple   at the end of an expression suffices:

Don't expand templates, even though you can.
If local variables are cheap and table lookups are expensive, then template expansion is way above your price bracket.

Avoid  like the plague. Nested template expansion using MediaWiki's preprocessor is what we're trying to get away from, after all. Most things that you'd do with that are done more simply, more quickly, and more maintainably, with simple Lua functions.

Similarly, avoid things like using w:Template:ISO 639 name aze (deleted August 2020) to store what is effectively an entry in a database. Reading it would be a nested parser call with concomitant database queries, all to map a string onto another string. Put a simple straightforward data table in your module, like the ones in Module:Language.