NaviServer - programmable web server

[ Main Table Of Contents | Table Of Contents | Keyword Index ]

tcl-overview(n) 5.0.0a manual "NaviServer Manual"

Name

tcl-overview - NaviServer Tcl Overview

Table Of Contents

Description

This chapter contains general information about using Tcl whether you are using the ADP approach or the Tcl Libraries approach, both of which use the same features of Tcl.

What is Tcl?

Tcl is a programming system developed by John Ousterhout at the University of California, Berkeley. According to Dr. Ousterhout:

Tcl is a simple scripting language for controlling and extending applications; its name stands for "tool command language". Tcl provides generic programming facilities, such as variables and loops and procedures, that are useful in a variety of applications. Furthermore, Tcl is embeddable. Its interpreter is a library of C procedures that can easily be incorporated into applications, and each application can extend the core Tcl features with additional commands for that application.

NaviServer supports the Tcl 8.* commands. When you write NaviServer extensions in Tcl, you use the core functionality of Tcl as well as a set of NaviServer-specific Tcl functions. For example, there is a NaviServer Tcl function called ns_conn that allows your Tcl script to obtain information about the current connection. In addition, functions are available for returning results to Web clients, accessing databases, and managing the permissions (access control) system.

This book describes the two different methods for extending NaviServer with Tcl, using NaviServer Dynamic Pages (Chapter 2) and Tcl libraries (see Chapter 3), plus general information on using Tcl with either method (see Chapter 4). It also provides a reference for NaviServer's Tcl API (see Chapter 5).

Recommended Reading

For more information on the Tcl language, we recommend the following sources:

  • Practical Programming in Tcl and Tk, by Brent B. Welch (Prentice Hall PTR, 1995)

  • Tcl and the Tk Toolkit, by John K. Ousterhout (Addison-Wesley, 1994)

  • For the complete manual pages for the Tcl v7.6 commands, visit the web site http://www.scriptics.com/man.

  • Philip Greenspun's web site, http://www-swiss.ai.mit.edu/wtr/dead-trees/, which includes chapters from his book, Database Backed Web Sites, The Thinking Person's Guide to Web Publishing, and numerous examples of NaviServer Tcl database applications.

  • The web site: http://www.yahoo.com/Computers_and_Internet/Programming_Languages/Tcl_Tk

Why Use Tcl?

You can extend NaviServer using Tcl, C, or CGI. However, using Tcl has the following advantages:

The C interface for NaviServer is described in the NaviServer C Developer's Guide, and the CGI interface is described in the NaviServer Administrator's Guide.

ADPs and Tcl Libraries

There are two ways to extend NaviServer using Tcl, and each is better-suited to different situations:

Tcl Interpreters

During NaviServer initialization, only one interpreter exists. While modules are loaded and initialized, they may add procedures to the interpreter. When initialization is complete (all modules are loaded and all Tcl libraries have been executed), NaviServer creates from this a blueprint consisting of the preloaded modules (mostly Tcl files). This blueprint is provided later to every thread (e.g. connection threads, or background threads) upon initialization.

NaviServer Blueprint

Figure 1: NaviServer Blueprint (per server)

Blueprints are used to reduce latency. When e.g. a connection thread processes an incoming request, all the basic Tcl code is already available and can be used by the interpreter without further loading operations.

In order to update the blueprints, NaviServer provides built-in support for updating via the command ns_eval or via the lower level command ns_ictl.

Global Variables

The NaviServer blueprint contains Tcl functions, XOTcl/NX classes and objects and also namespaced variables. This are all Tcl variables, which are not in the topmost namespace "::". The top-level variables are treated as "scratch variables", which are flushed automatically after every request. The global variables are often used for providing values for templating; cleaning these automatically can avoid unexpected interactions.

In addition of the per-interpreter variables, NaviServer provides shared variables, which can be accessed via the nsv_* interface (see below for an example). When such a variable is altered, its value is immediately available in all threads.

Sharing Files Between Interpreters

To share variables between the interpreters in a group, use the Tcl nsv_* command.

For example:

    #init time:
    nsv_set . sharedFile [open myfile.dat]

You must manually close the shared file (by performing a close) when it is no longer needed or when the server is shut down.

Note that there is no implicit locking between interpreters. As a consequence one interpreter can be performing a gets command on the shared file at the same time another interpreter performs a close on the same file. The results are unpredictable, and could potentially cause a core dump. To avoid this situation, use a mutex to protect access to the file whenever it is accessed.

For example:, at initialization time (for example, in your init.tcl script), open a shared file and create a mutex:

    nsv_set . sharedLock [ns_mutex create]
    nsv_set . sharedFile [open myfile.dat]

At run time, use the mutex to protect any actions you perform on the shared file:

    ns_mutex lock [nsv_get . sharedLock]
    set sharedFile  [nsv_get . sharedFile]
    puts $sharedFile ...
    gets $sharedFile ...
    ns_mutex unlock [nsv_get . sharedLock]

At shutdown (for example, in your shutdown procedure registered with ns_atshutdown), close the file and destroy the lock:

    close [nsv_get . sharedFile]
    ns_mutex destroy [nsv_get . sharedLock]

Working with ns_set and Form Data

An NaviServer operation procedure often manipulates sets of key-value pairs of string data, for example:

ns_set Data Structure

In the NaviServer C API these data are manipulated using the ns_set data structure. In the Tcl API the ns_set command can be used to manipulate underlying ns_set data structures that are generated by the NaviServer communications layer or the database services module.

The example below shows a typical use of the ns_set Tcl command to manipulate ns_set structures.

   # Example 2: Show header data
    #
    # Things to notice:
    # * The same function is registered for two different URLs
    # with different context.
    #
    # * The headers are pulled out of the conn using the
    # ns_conn function.
    #
    # * The value for a particular header line is extracted
    # with "ns_set iget" (case insensitive get).
 
    ns_register_proc GET /showbrowser showheader User-Agent
    ns_register_proc GET /showhost    showheader Host
 
    proc showheader {key} {
          set value [ns_set iget [ns_conn headers] $key]
          ns_return 200 text/plain "$key: $value"
    }

Tcl Examples

Example 1: hello

The following example script implements a simple request procedure which returns 'Hello World'.

 # Example 1: Hello World
 #
 # This simple operation just returns a plain text message.
 #
 # Things to notice:
 #
 # * ns_register_proc takes as arguments:
 #   * the HTTP method
 #   * the URL that the procedure handles
 #   * the procedure that is executed
 #
 # * ns_return takes as arguments:
 #   * the passed in connection
 #   * a return status, in this case 200 for success
 #   * a MIME type
 #   * the actual string to return
 #
 # * ns_return properly formats the HTTP response for you.
 
 ns_register_proc GET /example/hello hello
 
 proc hello {} {
   ns_return 200 text/plain "Hello World"
 }

The same script could also be implemented by specifying the script body directly in the ns_register_proc command invocation.

 ns_register_proc GET /example/hello {
   ns_return 200 text/plain "Hello World"
 }

The second variant is shorter, by defining procs, it might be easier for complex examples to e.g. update the procs at runtime, etc.

Example 2: showhdrs

The following example script shows how to access the HTTP headers sent by the client within a Tcl script.

 # Example 2: Show header data
 #
 # Things to notice:
 #
 # * The same function is registered for two different URLs
 #   with different context.
 #
 # * The headers are pulled out of the conn using the
 #   ns_conn function.
 #
 # * The value for a particular header line is extracted
 #   with "ns_set iget", the case insensitive counterpart to
 #   "ns_set get".
 
 ns_register_proc GET /showbrowser showheader User-Agent
 ns_register_proc GET /showhost    showheader Host
 
 proc showheader {key} {
    set value [ns_set iget [ns_conn headers] $key]
    ns_return 200 text/plain "$key: $value"
 }

Example 3a: genstory

The following example script provides two request procedures. The first procedure returns an HTML page for collecting a few fields of data from the user. The second procedure utilizes the data to generate a short story.

 # Example 3a: Form generation and handling
 #
 # Two functions are registered. One generates and
 # returns an HTML form, and the other processes
 # the data in the form.
 #
 # Things to notice:
 #
 # * Different functions are registered to the same
 #   URL with different methods.  Note that some browsers
 #   do not cache results properly when you do this.
 #
 # * The genstory function returns an error status
 #   (500) if the client doesn't pass in any form data.
 #
 # * Form data is stored in an ns_set, and accessed
 #   like any other set (e.g., header data).
 #
 # * A counter is used to loop through all the key
 #   value pairs in the form.
 
 ns_register_proc GET  /example/genstory genstoryform
 ns_register_proc POST /example/genstory genstory
 
 proc genstoryform {} {
  ns_return 200 text/html \
 "<html>
 <head>
 <title>Automatic Story Generator</title>
 </head>
 <body>
 <h1>Automatic Story Generator</h1>
 <form action='genstory' method='POST'>
 Noun: <input type='text' name='noun1'><br>
 Noun: <input type='text' name='noun2'><br>
 Name: <input type='text' name='name1'><br>
 Name: <input type='text' name='name2'><br>
 Adjective: <input type='text' name='adjective1'><br>
 Adjective: <input type='text' name='adjective2'><br>
 Verb: <input type='text' name='verb1'><br>
 Verb: <input type='text' name='verb2'><br>
 <p><input type='submit' value='Generate'>
 </form>
 </body></html>
 "}
 
 proc genstory {} {
  set formdata [ns_conn form]
 
  if {$formdata eq ""} {
    ns_return 200 text/plain "Need form data!"
    return
  }
 
  # Build up a human-readable representation of the form data.
 
  set hrformdata "<dl>"
  set size [ns_set size $formdata]
  for {set i 0} {$i < $size} {incr i} {
    append hrformdata "<dt>[ns_set key $formdata $i]</dt><dd>[ns_set value $formdata $i]</dd>"
  }
  append hrformdata "</dl>"
 
  ns_return 200 text/html \
 "<html>
 <head>
 <title>The story of [ns_set get $formdata name1] and
 [ns_set get $formdata name2]</title>
 </head>
 <body>
 <h1>The story of [ns_set get $formdata name1] and [ns_set get $formdata name2]</h1>
 <p>Once upon a time [ns_set get $formdata name1] and
 [ns_set get $formdata name2] went for a
 walk in the woods looking for a [ns_set get $formdata noun1].
 [ns_set get $formdata name1] was
 feeling [ns_set get $formdata adjective1] because
 [ns_set get $formdata name2] was so
 [ns_set get $formdata adjective2].  So
 [ns_set get $formdata name1] decided to
 [ns_set get $formdata verb1] [ns_set get $formdata name2]
 with a [ns_set get $formdata noun2].  This made
 [ns_set get $formdata name2] [ns_set get $formdata verb2]
 [ns_set get $formdata name1].
 <p><center>The End</center>
 The form data that made this possible:<br>
 $hrformdata
 </body></html>"
 }

Example 3b: pagetcl/genstory

The following example script implements the same story generating function of genstory (the previous example) but is implemented as a Tcl page instead of a library Tcl function. Save the following snippet as genstory.tcl as well as the associated HTML file genstory.html under the pages directory.

 # Example 3b: Form generation and handling
 #
 # This operation generates a story based on the
 # form data submitted from the form genstory.html.
 #
 # Things to notice:
 #
 # * This file should be stored with the HTML pages
 #   of the server.  When a client requests the URL corresponding
 #   to the file, the NaviServer sets the "conn" variable and
 #   evaluates the Tcl.
 #
 # * An error status (500) is returned if the client doesn't
 #   doesn't pass in any form data.
 #
 # * Form data is stored in an ns_set, and accessed
 #   like any other set (e.g., header data).
 #
 # * A counter is used to loop through all the key
 #   value pairs in the form.
 
 set formdata [ns_conn form]
 
 if {$formdata eq ""} {
    ns_return 200 text/plain "Need form data!"
    return
 }
 
 # Build up a human-readable representation of the form data.
 
 set hrformdata "<dl>"
 set size [ns_set size $formdata]
 for {set i 0} {$i < $size} {incr i} {
    append hrformdata "<dt>[ns_set key $formdata $i]</dt>\
        <dd>[ns_set value $formdata $i]</dd>"
 }
 append hrformdata "</dl>"
 
 ns_return 200 text/html \
 "<html>
 <head>
 <title>The story of [ns_set get $formdata name1] and
 [ns_set get $formdata name2]</title>
 </head>
 <body>
 <h1>The story of [ns_set get $formdata name1] and
 [ns_set get $formdata name2]</h1>
 <p>Once upon a time [ns_set get $formdata name1] and
 [ns_set get $formdata name2] went for a
 walk in the woods looking for a [ns_set get $formdata noun1].
 [ns_set get $formdata name1] was
 feeling [ns_set get $formdata adjective1] because
 [ns_set get $formdata name2] was so
 [ns_set get $formdata adjective2].  So
 [ns_set get $formdata name1] decided to
 [ns_set get $formdata verb1] [ns_set get $formdata name2]
 with a [ns_set get $formdata noun2].  This made
 [ns_set get $formdata name2] [ns_set get $formdata verb2]
 [ns_set get $formdata name1].
 <p><center>The End</center>
 The form data that made this possible:<br>
 $hrformdata
 </body></html>"

Here's the associated HTML file genstory.html:

 <html>
 <head>
 <title>Automatic Story Generator</title>
 </head>
 <body>
 <h1>Automatic Story Generator</h1>
 <form action='genstory.tcl' method='POST'>
 Noun: <input type='text' name='noun1'><br>
 Noun: <input type='text' name='noun2'><br>
 Name: <input type='text' name='name1'><br>
 Name: <input type='text' name='name2'><br>
 Adjective: <input type='text' name='adjective1'><br>
 Adjective: <input type='text' name='adjective2'><br>
 Verb: <input type='text' name='verb1'><br>
 Verb: <input type='text' name='verb2'><br>
 <p><input type='submit' value='Generate'>
 </form>
 </body>
 </html>

Example 4: redirect

The following example script shows how to use a NaviServer simple response command (in this case, ns_returnredirect) and the equivalent code when sending raw data to the client.

 # Example 4: Implementing redirects with ns_respond and
 # ns_write
 #
 # /example/not_here uses ns_respond to return an HTTP
 #   redirect to /example/finaldest.
 # /example/not_here2 does the same thing using ns_write
 # /example/not_here3 does the same thing with
 #   ns_returnredirect
 #
 # Things to notice:
 #
 # * When you use ns_write, you need to compose the
 #   entire response.
 #
 # * "ns_conn location" returns the http://hostname
 #   part of the URL that you can use to generate
 #   fully qualified URLs.
 #
 # * ns_returnredirect is a lot simpler than either
 #   ns_respond or ns_write.
 
 ns_register_proc GET /example/not_here  not_here
 ns_register_proc GET /example/not_here2 not_here2
 ns_register_proc GET /example/not_here3 not_here3
 ns_register_proc GET /example/finaldest finaldest
 
 proc not_here {} {
   ns_respond -status 302 -type text/plain \
      -string "Redirection" \
      -headers [ns_set new myheaders \
         Location [ns_conn location]/example/finaldest] \
         ]
 }
 
 proc not_here2 {} {
   set content [subst [ns_trim -delimiter | {
        |<html>
        |<head><title>Redirection</title></head>
        |<body>
        |<h1>Redirection</h1>
        |The actual location of what you were looking for is
        |<a href='[ns_conn location]/example/finaldest'> here.</a>
        |</body>
        |</html>}]]
 
   ns_write [subst [ns_trim -delimiter | {
        |HTTP/1.0 302 Document follows
        |MIME-Version: 1.0
        |Content-Type: text/html
        |Content-Length: [string length $content]
        |Location: [ns_conn location]/example/finaldest
        |
        |$content
        |}]]
 }
 
 proc not_here3 {} {
   ns_returnredirect [ns_conn location]/example/finaldest
 }
 
 proc finaldest {} {
   ns_return 200 text/plain \
     "You have arrived at the final destination."
 }

Example 5: desctable

The following example script provides a request procedure which describes the columns of a database table using the NaviServer "ns_tableinfo" command .

 # Example 5: Describing a database table
 #
 # /example/describetable prints out a column-by-column
 #   description of a database table.  The database
 #   pool name and table name are specified at the end
 #   of the URL -- e.g.,
 #
 #       /example/describetable/nsdbpool/ns_users
 #
 # Note: You must have the nsdb module loaded into your virtual
 #       server for this example to work.
 #
 # Things to notice:
 #
 # * ns_returnbadrequest returns a nicely formatted message
 #   telling the client they submitted an invalid request.
 #
 # * "ns_conn urlv" returns a Tcl array whose elements are the
 #   slash-delimited parts of the URL.
 #
 # * The describetable function loops through all the columns
 #   and uses "ns_column valuebyindex" to get the type of each
 #   one.
 #
 # * ns_returnnotice nicely formats the return value.
 
 ns_register_proc GET /example/describetable describetable
 
 proc describetable {} {
    if {[ns_conn urlc] != 4} {
        return [ns_returnbadrequest "Missing table name and/or poolname"]
    }
    set pool [lindex [ns_conn urlv] 2]
    if {$pool ni [ns_db pools]} {
        return [ns_returnbadrequest "Pool $pool does not exist"]
    }
    set db [ns_db gethandle $pool]
    set table [lindex [ns_conn urlv] 3]
    if {[info command ns_table] eq ""} {
       set output "Driver [ns_db driver $db] does not support command ns_table"
    } else {
        set tinfo [ns_table info $db $table]
        if {$tinfo eq ""} {
            return [ns_returnbadrequest "Table $table does not exist"]
        }
        set output "<dl>"
        set size [ns_column count $tinfo]
        for {set i 0} {$i < $size} {incr i} {
            append output "<dt>[ns_column name $tinfo $i] \
                    <dd>[ns_column typebyindex $tinfo $i]</dd>"
        }
        append output "</dl><hr>"
    }
    ns_returnnotice 200 "Table $table in pool $pool" $output
 }

Example 6: getempsm

The following example script shows how to query a table in the database.

 # Example 6: Getting data from the database
 #
 # /example/getemps queries a database in the default
 # pool and returns a list of all the employees listed
 # in the employees table.  It assumes a table called
 # employees exists with the column emp_name. 
 #
 # Note: You must have the nsdb module loaded into your virtual
 #       server for this example to work.
 #
 # Things to notice:
 #
 # * Use "ns_db gethandle" to get a handle for the database
 #   from the default database pool of the virtual server.
 #
 # * Use "ns_db select" to query the database and
 #   "ns_db getrow" to retrieve data.
 #
 # * Rows are returned as ns_sets.
 #
 
 ns_register_proc GET /example/getemps getemps
 
 proc getemps {} {
        set ul "<ul>"
        set db [ns_db gethandle [ns_config [ns_dbconfigpath] "DefaultPool"]]
        set row [ns_db select $db "select emp_name from employees order by emp_name;"]
        while { [ns_db getrow $db $row] } {
            append ul "<li>[ns_set get $row emp_name] \n"
        }
        append ul "</ul>"
        ns_returnnotice 200 "Employee list" $ul
 }
 

Example 7: wincgi

The following example script is a simple emulation of the WebSite WinCGI interface. Notice that this is just a demo, since the NaviServer modules nscgi provides a much simpler way to interface via CGI (when this is necessary).

 #
 # Example 7: simple emulation of the WebSite WinCGI interface
 #
 # This Tcl script emulates the WinCGI interface of the WebSite server.
 # To use, move this file to your Tcl library directory (normally the
 # modules/tcl directory of the NaviServer directory), set the
 # following nsd.ini variables in the [ns\server\<server-name>\wincgi]
 # section, and restart the server.
 #
 # key     default    description
 # ---     -------    -----------
 # prefix  /cgi-win   URL prefix for WinCGI.
 # debug   off        Set to on to keep temp files for debugging.
 # gmtoff  0          Minutes West of GMT for the "GMT Offset" variable.
 # dir     c:\wincgi  Directory of WinCGI programs.
 #
 #
 # Note:  This script is unsupported and not a complete emulation of the
 # WebSite WinCGI interface.  In particular, not all the WinCGI variables
 # are set.  Full support for WinCGI will be incorporated into the nscgi
 # module in a future NaviServer release.
 #
 
 #
 # Fetch the variables from the configuration file.
 #
 global WinCGI
 set WinCGI(section) "ns\\server\\[ns_info server]\\wincgi"
 if {[set WinCGI(prefix) [ns_config $WinCGI(section) prefix]] eq ""} {
    set WinCGI(prefix) /cgi-win
 }
 if {[set WinCGI(dir) [ns_config $WinCGI(section) dir]] eq ""} {
    set WinCGI(dir) [ns_info home]/$WinCGI(prefix)
 }
 if {[set WinCGI(gmtoff) [ns_config $WinCGI(section) gmtoff]] eq ""} {
    set WinCGI(gmtoff) 0
 }
 if {[set WinCGI(debug) [ns_config -bool $WinCGI(section) debug]] eq ""} {
    set WinCGI(debug) 0
 }
 
 
 #
 # Register the win-cgi procedure to handle requests for WinCGI executables.
 #
 ns_register_proc POST $WinCGI(prefix)/*.exe win-cgi
 ns_register_proc GET $WinCGI(prefix)/*.exe win-cgi
 
 
 #
 # win-cgi - The Tcl request procedure which emulates WinCGI.
 #
 proc win-cgi {} {
    global WinCGI
 
    # The program is the second part of the WinCGI URL.
    set args [join [split [ns_conn query] &]]
    set pgm [lindex [ns_conn urlv] 1]
    regsub -all {\+} $args " " args
    foreach e [split $WinCGI(dir)/$pgm /] {
            if {$e ne ""} {lappend exec $e}
    }
    set exec [join $exec ]
    if ![file executable $exec] {
            return [ns_returnnotfound]
    }
 
    # WinCGI requires a few temporary files.
    set ini [ns_mktemp]
    set inp [ns_mktemp]
    set out [ns_mktemp]
 
    # Copy the request content to the input file.
    set fp [open $inp w]
    ns_writecontent $fp
    set len [tell $fp]
    close $fp
 
    # Create the WinCGI variables .ini file.
    set fp [open $ini w]
    puts $fp {[CGI]}
    puts $fp \
 "Request Protocol=HTTP/1.0
 Request Method=[ns_conn method]
 Executable Path=$WinCGI(prefix)/$pgm
 Server Software=[ns_info name]/[ns_info version]
 Server Name=[ns_info name]
 Server Port=[ns_info version]
 Server Admin=[ns_config NaviServer WebMaster]
 CGI Version=CGI/1.2 (Win)
 Remote Address=[ns_conn peeraddr]
 Authentication Method=Basic
 Authentication Realm=[ns_conn location]
 Content Type=application/x-www-form-urlencoded
 Content Length=$len"
    puts $fp ""
    puts $fp {[System]}
    puts $fp \
 "GMT Offset=$WinCGI(gmtoff)
 Debug Mode=Yes
 Output File=$out
 Content File=$inp"
 
    # Set any POST or query form variables.
    puts $fp ""
    puts $fp {[Form Literal]}
    set form [ns_conn form]
    if {$form ne ""} {
            for {set i 0} {$i < [ns_set size $form]} {incr i} {
                    set key [ns_set key $form $i]
                    set value [ns_set value $form $i]
                    puts $fp "$key=$value"
            }
    }
 
    # Set the accept headers and accumulate the extra headers.
    puts $fp ""
    puts $fp {[Accept]}
    set headers [ns_conn headers]
    set extras ""
    for {set i 0} {$i < [ns_set size $headers]} {incr i} {
            set key [ns_set key $headers $i]
            set ukey [string toupper $key]
            set value [ns_set value $headers $i]
            if {$ukey eq "ACCEPT"} {
                    puts $fp "$value=Yes"
            } elseif {$key ne "CONTENT-LENGTH" && $key ne "CONTENT-TYPE"} {
                    append extras "$key=$value\n"
            }
    }
    puts $fp ""
    puts $fp {[Extra Headers]}
    puts $fp $extras
    close $fp
 
    # Execute the WinCGI program.
    # NB:  "catch" the exec and open because a WinCGI
    # program can be misbehaved, returning a nonzero
    # exit status or not creating the output file.
    catch {exec "$exec $ini $inp $out $args"}
    if [catch {set fp [open $out]}] {
            ns_returnerror 500 "WinCGI exec failed"
    } else {
            set type text/html
            set status 200
            while {[gets $fp line] > 0} {
                    set line [string trim $line]
                    if {$line eq ""} break
                    set head [split $line :]
                    set key [string tolower [string trim [lindex $head 0]]]
                    set value [string trim [lindex $head 1]]
                    if {$key eq "content-type"} {
                            set type $value
                    } elseif {$key eq "location"} {
                            set location $value
                    } elseif {$key eq "status"} {
                            set status $status
                    }
            }
            set page [read $fp]
            close $fp
 
            if [info exists location] {
                    ns_returnredirect $location
            } else {
                    ns_return $status $type $page
            }
    }
 
    if $WinCGI(debug) {
            ns_log Notice "CGI $pgm: ini: $ini, inp: $inp, out: $out"
    } else {
            file delete -- $ini
            file delete -- $inp
            file delete -- $out
    }
 }

Keywords

blueprint, configuration, loading, ns_set, nsdb, tcl-modules