webmcp

annotate doc/autodoc-header.htmlpart @ 520:ec1cca939315

Improved documentation of WebMCP's initialization procedure
author jbe
date Mon Aug 21 19:52:27 2017 +0200 (2017-08-21)
parents 55a43a67e9ad
children c5bf8d6963c4
rev   line source
jbe@509 1 <HTML>
jbe/bsw@0 2 <head>
jbe@381 3 <script type="text/javascript">
jbe@390 4 var expandedSections = {};
jbe@390 5 function toggleSection(ident) {
jbe@390 6 var element = document.getElementById('autodoc_details_' + ident);
jbe@390 7 if (element) {
jbe@390 8 if (expandedSections[ident]) {
jbe@390 9 element.style.display = 'none';
jbe@390 10 delete expandedSections[ident];
jbe@390 11 } else {
jbe@390 12 element.style.display = '';
jbe@390 13 expandedSections[ident] = true;
jbe@390 14 }
jbe@381 15 }
jbe@390 16 }
jbe@391 17 var inProgress = false;
jbe@390 18 window.onload = window.onhashchange = function() {
jbe@391 19 if (inProgress) return;
jbe@390 20 for (var ident in expandedSections) toggleSection(ident);
jbe@390 21 var hash = window.location.hash;
jbe@390 22 if (hash) {
jbe@390 23 toggleSection(hash.replace(/^#/, ''));
jbe@391 24 inProgress = true;
jbe@391 25 window.location.hash = hash; // required after collapsing/expanding
jbe@391 26 inProgress = false;
jbe@390 27 }
jbe@389 28 };
jbe@381 29 </script>
jbe/bsw@0 30 <style>
jbe/bsw@0 31 body {
jbe/bsw@0 32 font-family: "Liberation Sans", sans-serif;
jbe/bsw@0 33 font-size: 11pt;
jbe/bsw@0 34 padding-bottom: 5ex;
jbe/bsw@0 35 }
jbe/bsw@0 36 .warning {
jbe/bsw@0 37 color: #ff0000;
jbe/bsw@0 38 }
jbe/bsw@0 39 h1, h2 {
jbe/bsw@0 40 font-family: "Liberation Serif", Georgia, serif;
jbe/bsw@0 41 }
jbe/bsw@0 42 h2 {
jbe/bsw@0 43 margin-bottom: 0.3ex;
jbe/bsw@0 44 }
jbe/bsw@0 45 p {
jbe@507 46 margin-bottom: 1ex;
jbe/bsw@0 47 line-height: 130%;
jbe/bsw@0 48 }
jbe/bsw@0 49 tt, pre {
jbe/bsw@0 50 font-size: 10pt;
jbe/bsw@0 51 }
jbe/bsw@0 52 tt {
jbe/bsw@0 53 font-weight: bold;
jbe/bsw@0 54 white-space: nowrap;
jbe/bsw@0 55 }
jbe/bsw@0 56 .autodoc_entry {
jbe/bsw@0 57 margin-top: 1ex;
jbe/bsw@0 58 margin-bottom: 1ex;
jbe/bsw@0 59 }
jbe/bsw@0 60 .autodoc_comment_tail {
jbe/bsw@0 61 font-style: italic;
jbe/bsw@0 62 }
jbe/bsw@0 63 .autodoc_entry .short_synopsis {
jbe/bsw@0 64 cursor: pointer;
jbe/bsw@0 65 }
jbe@507 66 .AUTODOC_details {
jbe/bsw@0 67 padding-left: 1em;
jbe/bsw@0 68 padding-right: 1em;
jbe/bsw@0 69 border: 1px solid #777;
jbe/bsw@0 70 }
jbe/bsw@0 71 .autodoc_synopsis {
jbe/bsw@0 72 font-weight: bold;
jbe/bsw@0 73 }
jbe/bsw@0 74 .autodoc_synopsis .autodoc_comment_tail {
jbe/bsw@0 75 font-weight: normal;
jbe/bsw@0 76 color: #008000;
jbe/bsw@0 77 }
jbe/bsw@0 78 .autodoc_entry .autodoc_comment {
jbe/bsw@0 79 color: #400080;
jbe/bsw@0 80 }
jbe/bsw@0 81 .autodoc_source {
jbe/bsw@0 82 color: #505050;
jbe/bsw@0 83 }
jbe/bsw@0 84 </style>
jbe@446 85 <title>WebMCP 2.1.0 Documentation</title>
jbe/bsw@0 86 </head>
jbe/bsw@0 87 <body>
jbe@446 88 <h1>WebMCP 2.1.0 Documentation</h1>
jbe/bsw@0 89 <p>
jbe@278 90 WebMCP is a web development framework based on the Lua programming language (read more about Lua <a href="http://www.lua.org/about.html">here</a>).
jbe/bsw@0 91 </p>
jbe/bsw@0 92 <h2>Requirements</h2>
jbe/bsw@0 93 <p>
jbe@387 94 WebMCP has been developed on Linux and FreeBSD. Using it with Mac&nbsp;OS&nbsp;X is untested as of yet; Microsoft Windows is not supported. Beside the operating system, the only mandatory dependencies for WebMCP are the <a href="http://www.lua.org/">programming language Lua</a> version 5.2 or 5.3, the <a href="http://www.public-software-group.org/moonbridge">Moonbridge Network Server for Lua Applications</a> version 1.0.1 or higher, <a href="http://www.postgresql.org/">PostgreSQL</a> version 8.2 or higher, and a C compiler.
jbe/bsw@0 95 </p>
jbe/bsw@0 96 <h2>Installation</h2>
jbe/bsw@0 97 <p>
jbe@515 98 After downloading the tar.gz package, unpack it, enter the unpacked directory and type <tt>make</tt>. If you use Mac OS X or if you experience problems during compilation, you need to edit the <tt>Makefile.options</tt> file prior to compilation. The framework itself will be available in the <tt>framework/</tt> directory, while a demo application is available in the <tt>demo-app/</tt> directory. The <tt>framework.precompiled/</tt> and <tt>demo-app.precompiled/</tt> directories will contain a version with all Lua files being byte-code pre-compiled, which can be used instead. You may copy these directories (with <tt>cp -L</tt> to follow links) to any other place you like. Don't forget to setup a database, and make the <tt>tmp/</tt> directory of the application writable for the user who executes WebMCP. Good luck and have fun!
jbe@507 99 </p>
jbe@507 100 <h2>Configuration, initializers, and request handling</h2>
jbe@507 101 <p>
jbe@507 102 WebMCP uses the <a href="http://www.public-software-group.org/moonbridge">Moonbridge Network Server</a> to handle HTTP requests. The Moonbridge Network Server listens to a TCP port and passes control to WebMCP by calling <a href="#request.handler"><tt>request.handler(...)</tt></a> for each request. However, before each request is processed, WebMCP will initialize the environment. This initialization includes tasks such as
jbe@507 103 </p>
jbe@507 104 <ul>
jbe@507 105 <li>loading libraries,</li>
jbe@507 106 <li>configuring the database connection or the application,</li>
jbe@507 107 <li>connecting to the database,</li>
jbe@507 108 <li>etc.</li>
jbe@507 109 </ul>
jbe@507 110 <p>
jbe@507 111 For each request, it is also possible to execute filters. Filters can be used to
jbe@278 112 </p>
jbe@507 113 <ul>
jbe@507 114 <li>
jbe@507 115 restrict access for certain requests, e.g. by performing redirects or raising errors
jbe@507 116 </li>
jbe@507 117 <li>
jbe@507 118 perform tasks that are common for different requests, e.g.
jbe@507 119 <ul>
jbe@507 120 <li>execution certain database requests,</li>
jbe@507 121 <li>prepare a menu on the website,</li>
jbe@507 122 <li>&hellip;</li>
jbe@507 123 </ul>
jbe@507 124 </li>
jbe@507 125 </ul>
jbe@278 126 <p>
jbe@513 127 Filters and initializers are created by adding files in the application's directory structure. The filename determins the execution order of otherwise equally ranked initializers and/or filters. It is a common idiom to start the filename of a filter or initializer with a two digit number to be easily able to change the execution order when desired. Filters and initializers are executed both before and after a request. Each file must contain an <a href="#execute.inner"><tt>execute.inner()</tt></a> command. The part before that command is executed before the request, and the part after that command is executed after the request.
jbe@507 128 </p>
jbe@507 129 <p>
jbe@507 130 The Moonbridge server creates forks (i.e. clones) of the application server process (i.e. the whole Lua engine including all libraries and variables) in order to handle concurrent requests. Certain initializations may be performed before forking, other initializations must be performed after forking. For this purpose, WebMCP allows an application to provide so-called "pre-fork" and "post-fork" initializers. The application's configuration files as well as its pre-fork initializers are executed before forking. The application's post-fork initializers are executed after forking. In particular, any libraries that open file or network handles during initialization must not be loaded before the server process is forked. Opening database connections must be performed after forking as well. WebMCP follows the following execution order (directory structure is explained further down):
jbe@278 131 </p>
jbe@278 132 <ol>
jbe@278 133 <li>
jbe@520 134 Initialization code of <tt>mcp.lua</tt> in the <tt>bin/</tt> directory of the framework, including:
jbe@520 135 <ul>
jbe@520 136 <li>Initialization of WebMCP's autoloader in <tt>mcp.lua</tt> (in the <tt>bin/</tt> directory of the framework),</li>
jbe@520 137 <li>Execution of <tt>env/__init.lua</tt> (both relative to the framework and, if existent, relative to the application base directory)</li>
jbe@520 138 <li>Loading of most of WebMCP's libraries (excluding "multirand" library, which must not be loaded prior to forking);</li>
jbe@520 139 </ul>
jbe@278 140 </li>
jbe@278 141 <li>
jbe@520 142 Executing the selected configuration file: <tt>config/</tt><i>configuration_name</i><tt>.lua</tt>;
jbe@278 143 </li>
jbe@278 144 <li>
jbe@520 145 Executing all pre-fork initializers (both those in the <tt>app/_prefork/</tt> and those in the <tt>app/</tt><i>application_name</i><tt>/_prefork/</tt> directory) until call of <tt>execute.inner()</tt> within each initializer;
jbe@278 146 </li>
jbe@278 147 <li>
jbe@520 148 The Moonbridge Network Server forks the process (i.e. cloning the whole Lua machine);<br />
jbe@278 149 <span style="color: red">Note: no file handles or network connections must be opened prior to this point!</span>
jbe@278 150 </li>
jbe@278 151 <li>
jbe@520 152 Loading WebMCP's "multirand" library;
jbe@281 153 </li>
jbe@281 154 <li>
jbe@520 155 Executing all post-fork initializers (both those in the <tt>app/_postfork/</tt> and those in the <tt>app/</tt><i>application_name</i><tt>/_postfork/</tt> directory) until call of <tt>execute.inner()</tt> within each initializer;
jbe@278 156 </li>
jbe@278 157 <li>
jbe@278 158 For each request:
jbe@278 159 <ul>
jbe@278 160 <li>
jbe@520 161 Execution of all applicable filters until call of <tt>execute.inner()</tt> within each filter,
jbe@278 162 </li>
jbe@278 163 <li>
jbe@520 164 Handling of the request by calling the appropriate view or action,
jbe@278 165 </li>
jbe@278 166 <li>
jbe@520 167 Resuming execution of all filters in reverse order from that position where <tt>execute.inner()</tt> had been called;
jbe@278 168 </li>
jbe@278 169 </ul>
jbe@278 170 </li>
jbe@278 171 <li>
jbe@520 172 Resuming execution of all post-fork initializers in reverse order from that position where <tt>execute.inner()</tt> had been called;
jbe@278 173 </li>
jbe@278 174 <li>
jbe@520 175 Resuming execution of all pre-fork initializers in reverse order from that position where <tt>execute.inner()</tt> had been called.
jbe@278 176 </li>
jbe@278 177 </ol>
jbe/bsw@0 178 </p>
jbe@439 179 <p>
jbe@509 180 As a minimum configuration, the used configuration file or pre-fork initializer should at least contain a <a href="#listen"><tt>listen{...}</tt></a> call, e.g.:
jbe@439 181 </p>
jbe@439 182 <pre>
jbe@439 183 listen{
jbe@439 184 { proto = "tcp", host = "::", port = 8080 },
jbe@439 185 { proto = "tcp", host = "0.0.0.0", port = 8080 }
jbe@439 186 }
jbe@507 187 execute.inner() -- only use this line if done in pre-fork initializer</pre>
jbe/bsw@0 188 <h2>Using the atom library</h2>
jbe/bsw@0 189 <p>
jbe/bsw@0 190 Lua itself has only very few built-in data types. The atom library gives support for extra data types. Currently the following extra data types are provided:
jbe/bsw@0 191 </p>
jbe/bsw@0 192 <ul>
jbe@509 193 <li><a href="#atom.fraction:new">atom.fraction</a></li>
jbe@509 194 <li><a href="#atom.date:new">atom.date</a></li>
jbe@509 195 <li><a href="#atom.time:new">atom.time</a></li>
jbe@509 196 <li><a href="#atom.timestamp:new">atom.timestamp (date and time combined in one data type)</a></li>
jbe/bsw@0 197 </ul>
jbe/bsw@0 198 <p>
jbe/bsw@0 199 In addition the following pseudo-types are existent, corresponding to Lua's base types:
jbe/bsw@0 200 </p>
jbe/bsw@0 201 <ul>
jbe/bsw@0 202 <li>atom.boolean</li>
jbe/bsw@0 203 <li>atom.string</li>
jbe/bsw@0 204 <li>atom.integer</li>
jbe/bsw@0 205 <li>atom.number</li>
jbe/bsw@0 206 </ul>
jbe/bsw@0 207 <p>
jbe/bsw@0 208 Both atom.integer and atom.number refer to Lua's base type &ldquo;number&rdquo;.
jbe/bsw@0 209 </p>
jbe/bsw@0 210 <p>
jbe@514 211 New values of atom data types are created by either calling <tt>atom.</tt><i>type</i><tt>:load(string_representation)</tt> or by calling <tt>atom.</tt><i>type</i><tt>{...}</tt>, e.g. <tt>atom.date{year=1970, month=1, day=1}</tt>. Note that <tt>atom.date{...}</tt> is a shortcut for <tt><a href="#atom.date:new">atom.date:new</a>{...}</tt>. You can dump any atom value as a string by calling <a href="#atom.dump"><tt>atom.dump(value)</tt></a> and later reload it with <tt>atom.</tt><i>type</i><tt>:load(string)</tt>.
jbe/bsw@0 212 </p>
jbe/bsw@0 213 <h2>Using the Object-Relational Mapper &ldquo;mondelefant&rdquo;</h2>
jbe/bsw@0 214 <p>
jbe@278 215 The library &ldquo;mondelefant&rdquo; shipping with WebMCP can be used to access PostgreSQL databases. It also serves as an Object-Relational Mapper (ORM). The database connection is usually configured in the config file (e.g. in <tt>config/devel.lua</tt>):
jbe@278 216 </p>
jbe@278 217 <pre>
jbe@439 218 config.db = { engine="postgresql", dbname="webmcp_demo" }</pre>
jbe@278 219 <p>
jbe@278 220 In addition to configuring the database, it must be opened within a post-fork initializer (e.g. in <tt>app/_postfork/01_database.lua</tt>):
jbe/bsw@0 221 </p>
jbe/bsw@0 222 <pre>
jbe@509 223 _G.db = assert(<a href="#mondelefant.connect">mondelefant.connect</a>(config.db))
jbe@509 224 function mondelefant.class_prototype:<a href="#db_class:get_db_conn">get_db_conn</a>() return db end
jbe@439 225 execute.inner()</pre>
jbe/bsw@0 226 <p>
jbe@515 227 The parameters for <tt>mondelefant.connect</tt> are directly passed to PostgreSQL's client library libpq. See PostgreSQL's documentation for information about <a href="https://www.postgresql.org/docs/9.6/static/libpq-connect.html#LIBPQ-PARAMKEYWORDS">supported parameters</a>.
jbe/bsw@0 228 </p>
jbe/bsw@0 229 <p>
jbe/bsw@0 230 To define a model to be used within a WebMCP application, create a file named with the name of the model and <tt>.lua</tt> as extension in the <tt>model/</tt> directory of your application. The most basic definition of a model (named &ldquo;movie&rdquo; in this example) is:
jbe/bsw@0 231 </p>
jbe/bsw@0 232 <pre>
jbe@509 233 Movie = <a href="#mondelefant.new_class">mondelefant.new_class</a>()
jbe/bsw@0 234 Movie.table = 'movie'</pre>
jbe/bsw@0 235 <p>
jbe/bsw@0 236 Note: Model classes are always written CamelCase, while the name of the file in <tt>model/</tt> is written lower_case.
jbe/bsw@0 237 </p>
jbe/bsw@0 238 <p>
jbe/bsw@0 239 To select objects from the database, the mondelefant library provides a selector framework:
jbe/bsw@0 240 </p>
jbe/bsw@0 241 <pre>
jbe@509 242 local s = Movie:<a href="#db_class:new_selector">new_selector</a>()
jbe@509 243 s:<a href="#db_selector:add_where">add_where</a>{ 'id = ?', param.get_id() }
jbe@509 244 s:<a href="#db_selector:single_object_mode">single_object_mode</a>() -- return single object instead of list
jbe@509 245 local movie = s:<a href="#db_selector:exec">exec</a>()</pre>
jbe/bsw@0 246 <p>
jbe/bsw@0 247 A short form of the above query would be:
jbe/bsw@0 248 </p>
jbe/bsw@0 249 <pre>
jbe/bsw@0 250 local movie = Movie:new_selector():add_where{ 'id = ?', param.get_id() }:single_object_mode():exec()</pre>
jbe/bsw@0 251 <p>
jbe/bsw@0 252 For more examples about how to use the model system, please take a look at the demo application.
jbe/bsw@0 253 </p>
jbe/bsw@0 254 <h2>The Model-View-Action (MVA) concept</h2>
jbe/bsw@0 255 <p>
jbe/bsw@0 256 As opposed to other web application frameworks, WebMCP does not use a Model-View-Controller (MVC) concept, but a Model-View-Action (MVA) concept.
jbe/bsw@0 257 </p>
jbe/bsw@0 258 <h3>Models</h3>
jbe/bsw@0 259 <p>
jbe@513 260 The models in MVA are like the models in MVC; they are used to access data, which is stored in a relational database (PostgreSQL), in an object oriented way. Methods provided by the corresponding classes can be used to alter stored objects or execute any other associated program code. Models are usually defined in a file with a lowercase filename ending with "<tt>.lua</tt>" in the <tt>models/</tt> directory of the application. The corresponding model name (i.e. class name) must be written in CamelCase, e.g. "<tt>models/my_model.lua</tt>" should define a model class named "<tt>MyModel</tt>". The simplest model is created by calling <a href="#mondelefant.new_class"><tt>mondelefant.new_class()</tt></a> and subsequently setting the <a href="#db_class.table"><tt>table</tt></a> attribute of the returned class.
jbe@509 261 </p>
jbe@509 262 <pre>
jbe@509 263 -- filename: model/customer_receipt.lua
jbe@509 264 CustomerReceipt = mondelefant.new_class()
jbe@509 265 CustomerReceipt.table = "custreceipt"</pre>
jbe@509 266 <p>
jbe@509 267 Methods such as <a href="#db_class:add_reference"><tt>:add_reference(...)</tt></a> can be used to further modify or extend the class.
jbe/bsw@0 268 </p>
jbe/bsw@0 269 <h3>Views</h3>
jbe/bsw@0 270 <p>
jbe@513 271 The views in the MVA concept are different from the views in the MVC concept. As WebMCP has no controllers, the views are responsible for processing the GET/POST parameters from the webbrowser, fetching the data to be displayed, and creating the output by directly writing HTML to slots of a layout (see <a href="#slot.select"><tt>slot.select(...)</tt></a>, <a href="#slot.put"><tt>slot.put(...)</tt></a>, and <a href="#slot.put_into"><tt>slot.put_into(...)</tt></a> or by calling helper functions for the user interface (those functions beginning with "<tt>ui.</tt>"). Views are stored in files with the file path "<tt>app/</tt><i>application_name</i><tt>/</tt><i>module_name</i><tt>/</tt><i>view_name</i><tt>.lua</tt>". When their corresponding URL, e.g. "<tt>http://</tt><i>hostname</i><tt>:</tt><i>port</i><tt>/</tt><i>module_name</i><tt>/</tt><i>view_name</i><tt>.html</tt>", is requested, the code in that file gets executed (after calling appropriate filters). After the execution of the view has finished (and after all filters have finished their execution too), the slot data will be inserted into placeholder sections in the selected layout file. The layout file defaults to "<tt>app/</tt><i>application_name</i><tt>/_layout/default.html</tt>" but may be changed using the <a href="#slot.set_layout"><tt>slot.set_layout(...)</tt></a> function.
jbe/bsw@0 272 </p>
jbe/bsw@0 273 <h3>Actions</h3>
jbe/bsw@0 274 <p>
jbe@509 275 Actions are similar to views, but supposed to change data in the database, hence only callable by HTTP POST requests. They are also responsible for processing the POST parameters from the webbrowser. They can modify the database, but instead of rendering a page to be displayed, they just return a status code string (via Lua's <tt>return</tt> statement, where <tt>true</tt> can also be used instead of "<tt>ok</tt>", and <tt>false</tt> instead of "<tt>error</tt>"). Depending on the status string there will be an internal forward or an HTTP 303 redirect to a view. When calling an action via a POST request, additional POST parameters, which are usually added by hidden form fields, determine the view to be displayed for each status string returned by the action. See the <tt>routing</tt> parameter to the <a href="#ui.form"><tt>ui.form{...}</tt></a> function for further details.
jbe/bsw@0 276 </p>
jbe@510 277 <h2>Layouts</h2>
jbe@510 278 <p>
jbe@510 279 Templates for HTML documents to be returned by views are stored at the path "<tt>app/</tt><i>application_name</i><tt>/_layout/</tt><i>layout_name</i><tt>.html</tt>", relative to the application base path. The default layout name is "<tt>default</tt>". For system errors, the layout name "<tt>system_error</tt>" is used. A sample layout is given as follows:
jbe@510 280 </p>
jbe@510 281 <pre>
jbe@510 282 &lt;!DOCTYPE HTML&gt;
jbe@510 283 &lt;html&gt;
jbe@510 284 &lt;head&gt;
jbe@510 285 &lt;title&gt;&lt!-- WEBMCP SLOTNODIV title --&gt;&lt;/title&gt;
jbe@510 286 &lt;link rel="stylesheet" type="text/css" media="screen" href="__BASEURL__/static/style.css"/&gt;
jbe@510 287 &lt;/head&gt;
jbe@510 288 &lt;body&gt;
jbe@510 289 &lt;!-- WEBMCP SLOT content --&gt;
jbe@510 290 &lt;/body&gt;
jbe@510 291 &lt;/html&gt;</pre>
jbe@510 292 <p>
jbe@510 293 The following elements in a layout file get replaced automatically:
jbe@510 294 </p>
jbe@510 295 <ul>
jbe@510 296 <li><tt>__BASEURL__</tt> gets replaced with the application's base web address.</li>
jbe@510 297 <li><tt>&lt;!-- WEBMCP SLOT </tt><i>slot_name</i><tt> --&gt;</tt> gets, if the slot is not empty, replaced with a <tt>&lt;div&gt;</tt> element (with the <tt>id</tt> attribute set to "<tt>slot_</tt><i>slotname</i>") that contains the slot's content.</li>
jbe@510 298 <li><tt>&lt;!-- WEBMCP SLOTNODIV </tt><i>slot_name</i><tt> --&gt;</tt> gets replaced with the slot's content (without wrapping it in an additional <tt>&lt;div&gt;</tt> element).</li>
jbe@510 299 </ul>
jbe@511 300 <h2>Internationalization/Localization</h2>
jbe@511 301 <p>
jbe@511 302 To translate certain strings in your application, simply use the <a href="#_">underscore function</a>. A language can be selected with <tt><a href="#locale.set">locale.set</a>{lang = "code"}</tt> where <tt>code</tt> is a language code. The translations for strings are expected to be contained in files "<tt>locale/translations.</tt><i>language_code</i><tt>.lua</tt>". These files should return a table that maps strings to their corresponding translation:
jbe@511 303 </p>
jbe@511 304 <pre>
jbe@511 305 return{
jbe@511 306 ["Are you sure?"] = "Bist Du sicher?";
jbe@511 307 ["User '#{name}' created"] = "Benutzer '#{name}' created";
jbe@511 308 }</pre>
jbe@511 309 <p>
jbe@511 310 Such translation files can be automatically created with the <tt>langtool.lua</tt> program, found in the framework's <tt>bin/</tt> directory.
jbe@511 311 </p>
jbe@518 312 <h2>Global variables and the environment</h2>
jbe@518 313 <p>
jbe@518 314 To avoid accidental programming errors, WebMCP forbids setting of global variables by default. This is overridden by using the prefix "<tt>_G.</tt>" (<a href="#_G">see reference</a>) when setting the variable, e.g. <tt>_G.myvar = 7</tt>, or by setting the variable in a file with the same name of the global varaible (but suffixed with "<tt>.lua</tt>") in the <tt>env/</tt> directory of the framework or application. Note, however, that the lifetime of global variables is not predictable as it depends on process recycling of the Moonbridge webserver (one fork will handle more than one request) and because there may be multiple forks of the Lua machine which all have their own global variable space (there is usually more than one fork).
jbe@518 315 </p>
jbe@518 316 <p>
jbe@519 317 If an application needs to store request related data, the global table <a href="#app"><tt>app</tt></a> should be used (e.g. <tt>app.myvar = true</tt> instead of <tt>_G.myvar = true</tt>). The <tt>app</tt> table gets automatically initialized (i.e. emptied) for each request.
jbe@519 318 </p>
jbe@519 319 <p>
jbe@518 320 Global variables are still useful when providing access to libraries, for example. WebMCP automatically loads certain libraries and functions though an autoloader mechanism. On read-accessing any unknown variable, WebMCP will search the framework's and application's <tt>env/</tt> directories for an appropriate file (e.g. "<tt>my_func.lua</tt> if you invoke "<tt>my_func()</tt>") or a matching directory (e.g. "<tt>my_module/</tt> if you access "<tt>my_module.my_func()</tt>). In case of an existing directory in <tt>env/</tt>, an empty table with autoloading capabilities is automatically created as global variable with the name of the directory. The autoloading mechanism allows directories to contain further files which can be used to initialize variables within that table when accessed. Directories can also contain a special file called "<tt>__init.lua</tt>" that always gets executed when the table is accessed for the first time. The <tt>env/</tt> root directory can also contain a file ("<tt>env/__init__.lua</tt>") which gets executed before any configuration is loaded.
jbe@518 321 </p>
jbe@518 322 <p>
jbe@519 323 A good place to store utility functions is a global table called <tt>util</tt>. This table will be automatically accessible if you create a <tt>env/util/</tt> directory in your WebMCP application. To provide a function <tt>util.myfunc(...)</tt> simply create a file <tt>env/util/myfunc.lua</tt>, with the following function definition:
jbe@518 324 </p>
jbe@519 325 <pre>
jbe@519 326 -- filename: env/util/myfunc.lua
jbe@519 327 function util.myfunc()
jbe@519 328 slot.put_into("hello", "Hello World!")
jbe@519 329 end</pre>
jbe/bsw@0 330 <h2>Directory structure of a WebMCP application</h2>
jbe@513 331 <p>
jbe@513 332 Summarizing information from the previous section, we get the following directory structure for a WebMCP application:
jbe@513 333 </p>
jbe/bsw@0 334 <ul>
jbe/bsw@0 335 <li>
jbe/bsw@0 336 Base Directory
jbe/bsw@0 337 <ul>
jbe/bsw@0 338 <li>
jbe/bsw@0 339 <tt>app/</tt>
jbe/bsw@0 340 <ul>
jbe@510 341 <li><tt>_prefork/</tt></li>
jbe@510 342 <li><tt>_postfork/</tt></li>
jbe/bsw@0 343 <li>
jbe@510 344 <tt>main/</tt> (typically "main" is the only application)
jbe@278 345 <ul>
jbe@510 346 <li><tt>_prefork/</tt></li>
jbe@510 347 <li><tt>_postfork/</tt></li>
jbe@510 348 <li><tt>_filter/</tt></li>
jbe@510 349 <li><tt>_filter_action/</tt></li>
jbe@510 350 <li><tt>_filter_view/</tt></li>
jbe@510 351 <li><tt>_layout/</tt></li>
jbe@278 352 <li>
jbe@510 353 <tt>index/</tt> (module "index" is the default module)
jbe/bsw@0 354 <ul>
jbe/bsw@0 355 <li>
jbe/bsw@0 356 <tt>_action/</tt>
jbe/bsw@0 357 <ul>
jbe@510 358 <li><i>action_name</i><tt>.lua</tt></li>
jbe@510 359 <li><i>another_action_name</i><tt>.lua</tt></li>
jbe/bsw@0 360 <li>&hellip;</li>
jbe/bsw@0 361 </ul>
jbe/bsw@0 362 </li>
jbe@510 363 <li><tt>index.lua</tt> (view "index" of module "index" is the default view)</li>
jbe@510 364 <li><i>other_view_name</i><tt>.lua</tt></li>
jbe/bsw@0 365 <li>&hellip;</li>
jbe/bsw@0 366 </ul>
jbe/bsw@0 367 </li>
jbe/bsw@0 368 <li>
jbe/bsw@0 369 <i>other_module_name</i><tt>/</tt>
jbe@510 370 <ul><li>&hellip;</li></ul>
jbe/bsw@0 371 </li>
jbe/bsw@0 372 </ul>
jbe/bsw@0 373 </li>
jbe/bsw@0 374 <li>
jbe/bsw@0 375 <i>other_application_name</i><tt>/</tt>
jbe/bsw@0 376 <ul>
jbe/bsw@0 377 <li>&hellip;</li>
jbe/bsw@0 378 </ul>
jbe/bsw@0 379 </li>
jbe/bsw@0 380 </ul>
jbe/bsw@0 381 </li>
jbe/bsw@0 382 <li>
jbe/bsw@0 383 <tt>config/</tt>
jbe/bsw@0 384 <ul>
jbe@510 385 <li><tt>development.lua</tt> (e.g. to be used for development purposes)</li>
jbe@510 386 <li><tt>production.lua</tt> (e.g. to be used for production)</li>
jbe@510 387 <li><i>other_config_name</i><tt>.lua</tt></li>
jbe/bsw@0 388 <li>&hellip;</li>
jbe/bsw@0 389 </ul>
jbe/bsw@0 390 </li>
jbe/bsw@0 391 <li>
jbe/bsw@0 392 <tt>db/</tt>
jbe/bsw@0 393 <ul>
jbe@510 394 <li><tt>schema.sql</tt> (not used by WebMCP, but may be used to initialize a database)</li>
jbe/bsw@0 395 </ul>
jbe/bsw@0 396 </li>
jbe/bsw@0 397 <li>
jbe@518 398 <tt>env/</tt>
jbe@518 399 <ul>
jbe@518 400 <li><tt>__init.lua</tt></li>
jbe@519 401 <li><i>some_global_variable_name</i><tt>.lua</tt></li>
jbe@518 402 <li>
jbe@519 403 <i>another_global_variable_name</i><tt>/</tt>
jbe@518 404 <ul>
jbe@518 405 <li><tt>__init.lua</tt></li>
jbe@518 406 <li><i>attribute_name</i><tt>.lua</tt></li>
jbe@518 407 <li>
jbe@518 408 <i>another_attibute_name</i><tt>/</tt>
jbe@518 409 <ul><li>&hellip;</li></ul>
jbe@518 410 </li>
jbe@518 411 </ul>
jbe@518 412 </li>
jbe@518 413 </ul>
jbe@518 414 <li>
jbe@510 415 <tt>locale/</tt> (translation files used by the <a href="#_">underscore function</a>")
jbe/bsw@0 416 <ul>
jbe/bsw@0 417 <li>
jbe/bsw@0 418 <tt>translations.de.lua</tt>
jbe/bsw@0 419 </li>
jbe/bsw@0 420 <li>
jbe/bsw@0 421 <tt>translations.en.lua</tt>
jbe/bsw@0 422 </li>
jbe/bsw@0 423 <li>
jbe/bsw@0 424 <tt>translations.</tt><i>languagecode</i><tt>.lua</tt>
jbe/bsw@0 425 </li>
jbe/bsw@0 426 <li>&hellip;</li>
jbe/bsw@0 427 </ul>
jbe/bsw@0 428 </li>
jbe/bsw@0 429 <li>
jbe/bsw@0 430 <tt>model/</tt>
jbe/bsw@0 431 <ul>
jbe/bsw@0 432 <li>
jbe@510 433 <i>model_name</i><tt>.lua</tt> (code must create a model class in camel case, e.g. "<tt>ModelName</tt>" if the file name is "<tt>model_name.lua</tt>")
jbe/bsw@0 434 </li>
jbe/bsw@0 435 <li>
jbe/bsw@0 436 <i>another_model_name</i><tt>.lua</tt>
jbe/bsw@0 437 </li>
jbe/bsw@0 438 <li>&hellip;</li>
jbe/bsw@0 439 </ul>
jbe/bsw@0 440 </li>
jbe/bsw@0 441 <li>
jbe/bsw@0 442 <tt>static/</tt>
jbe/bsw@0 443 <ul>
jbe/bsw@0 444 <li>&hellip; (images, javascript, ...)</li>
jbe/bsw@0 445 </ul>
jbe/bsw@0 446 </li>
jbe/bsw@0 447 <li>
jbe@510 448 <tt>tmp/</tt> (needs to be writable by the user executing WebMCP)
jbe/bsw@0 449 </li>
jbe/bsw@0 450 </ul>
jbe/bsw@0 451 </li>
jbe/bsw@0 452 </ul>
jbe@278 453 <h2>Starting your application</h2>
jbe@278 454 <p>
jbe@513 455 Ensure that the <tt>moonbridge</tt> binary is within your system's search path and that the <tt>moonbridge_http.lua</tt> file is included in the LUA_PATH or linked into the framework's <tt>lib/</tt> directory (alternatively the MOONBR_LUA_PATH option might be set accordingly at compile-time of the Moonbridge Network Server). To start an application, call the <tt>mcp.lua</tt> executable (found in <tt>framework/bin/mcp.lua</tt>) with the following four arguments:
jbe@513 456 </p>
jbe@513 457 <ol>
jbe@513 458 <li>Path of the WebMCP framework directory, e.g. <tt>./framework</tt></li>
jbe@513 459 <li>Path of your application's directory, e.g. <tt>./demo-app</tt></li>
jbe@513 460 <li>Name of your applicaiton (usually <tt>main</tt>)</li>
jbe@513 461 <li>Name of configuration (e.g. <tt>devel</tt> to use config/devel.lua)</li>
jbe@513 462 </ol>
jbe@513 463 <p>
jbe@513 464 Alternatively, the <tt>moonbridge</tt> binary may be called directly with the following five arguments:
jbe@278 465 </p>
jbe@278 466 <ol>
jbe@513 467 <li>Path to <tt>mcp.lua</tt>, e.g. <tt>./framework/bin/mcp.lua</tt></li>
jbe@513 468 <li>Path of the WebMCP framework directory, e.g. <tt>./framework</tt></li>
jbe@513 469 <li>Path of your application's directory, e.g. <tt>./demo-app</tt></li>
jbe@513 470 <li>Name of your applicaiton (usually <tt>main</tt>)</li>
jbe@513 471 <li>Name of configuration (e.g. <tt>devel</tt> to use config/devel.lua)</li>
jbe@278 472 </ol>
jbe@513 473 <p>
jbe@513 474 Note that the demo application will require a database to be set up prior to starting. Execute the following shell commands first:
jbe@513 475 </p>
jbe@513 476 <pre>
jbe@513 477 createdb webmcp_demo
jbe@513 478 psql -v ON_ERROR_STOP=1 -f demo-app/db/schema.sql webmcp_demo</pre>
jbe/bsw@0 479 <h2>Automatically generated reference for the WebMCP environment</h2>
jbe/bsw@0 480 <ul>

Impressum / About Us