Forum OpenACS Development: ns_job valid procs

Collapse
Posted by Jonathan Lee on
I was experimenting with aolserver's ns_job and noticed there are limitations to what I can include in the passed script. I can pass OpenACS procs like db_0or1row, but I can't pass custom ad_procs that I have defined in the same .tcl file. Is there some step needed to pre-load these procs?

For example, if I define the following, the ns_job piece fails for not recognizing "foo":

ad_proc foo {} {
    puts "hello, world"
}

ad_proc test_ns_jobs {} {
    ns_job queue test_queue {
        foo
    }
}

if {test_queue ni [ns_job queues]} {
    ns_job create test_queue
}

test_ns_jobs

Collapse
2: Re: ns_job valid procs (response to 1)
Posted by Gustaf Neumann on

If you simply define the proc in a script to be executed, the proc is defined in the current interpreter only. To ensure that new procedures are incorporated into the blueprints for all threads/interpreters (including in the interpreter used in the job queue), you must define them using ns_eval (see man page).

However, when a proc is defined in the library files, which are loaded at startup time (as in OpenACS under "packages"), the proc becomes part of the blueprints for all future interpreters, and there is no need for ns_eval.

Here is your example slightly modified to use ns_eval:

ns_eval {
  ad_proc foo {} {
    ns_log warning "hello, world"
  }
}

if {"test_queue" ni [ns_job queues]} {
    ns_job create test_queue
}
ns_job queue test_queue foo

This example works with NaviServer (AOLserver is now obsolete).

Note: Avoid using puts to output to stdout, as this output can be difficult to locate. Instead, use ns_log for logging messages.

Hope this helps!
-g

Collapse
3: Re: ns_job valid procs (response to 2)
Posted by Jonathan Lee on
Thanks!
I am unfortunately tasked with adding some functionality to an old application stack, so I hesitate to migrate from AOLserver to NaviServer without bandwidth to retest all the features. I found that this worked in my local test environment with NaviServer but not in the existing staging environment with AOLserver 4.5:

ns_eval {
ad_proc foo {bar} {
ns_log warning "hello, $bar"
}
}

ns_job queue test_queue foo "world"

I get an error "unknown variable $bar". I'm not sure if this was a limitation in AOLserver that has since been fixed in NaviServer or if there's something else at work here. Regardless, I'm able to queue library-defined procs that don't take arguments, which was my greatest need.

Collapse
4: Re: ns_job valid procs (response to 3)
Posted by Gustaf Neumann on

The NaviServer does not produce the error with variable "bar", just tested the following snippet.

ns_eval {
  ad_proc foo {bar} {
    ns_log notice "hello, $bar"
  }
}

if {"test_queue" ni [ns_job queues]} {
    ns_job create test_queue
}
ns_job queue test_queue [list foo XXX]

Notice, that ns_job expects the script as a single argument.

all the best
-g

Collapse
5: Re: ns_job valid procs (response to 3)
Posted by Brian Fenton on
Hi Jonathan

do regular Tcl procs defined with "proc" (as opposed to those defined with "ad_proc" )work for you?

best wishes
Brian

Collapse
6: Re: ns_job valid procs (response to 5)
Posted by Jonathan Lee on

It's the weirdest thing. I know AOLserver supports ns_eval, but in the folder I was given to prototype new features, I get errors simply by having variables in the ns_eval code block:

can't read "bar": no such variable
    while executing
"ns_eval {
  proc foo {bar} {
    ns_log notice "hello, $bar"
  }
}"

Now I can get around this by relying on adprocs that are loaded when the server starts, but then I still can't seem to get ns_job to function the way I'd expect. If foo is already defined (in this case, as an adproc), I don't get a runtime error with this in a tcl script loaded in tandem with an adp page:

set bar "world"
ns_job queue test_queue [foo $bar]

but if I review the queue's job list, I see an empty string: id job36 state done results {} script {} code TCL_OK type nondetached req none time 0.01 starttime 1743162229 endtime 1743162229

indicating the foo proc was executed before anything was added to the queue. If use curly braces:

set bar "world"
ns_job queue test_queue {[foo $bar]}

I get the following in the queue: id job38 state done results {can't read "bar": no such variable} script {[foo $bar]} code TCL_ERROR type nondetached req none time 0.06 starttime 1743163464 endtime 1743163464

I could try again with standard procs, but this feels like something wrong with how my script is being evaluated.

Collapse
7: Re: ns_job valid procs (response to 6)
Posted by Brian Fenton on
I found an AOLserver instance and the following seems to work for me. Notice the double curly brackets in the ns_eval, which I got from here https://panoptic.com/wiki/aolserver/Ns_eval

ns_eval {{proc foo bar {ns_log Notice bar=$bar}}}
if {"test_queue" ni [ns_job queues]} {
ns_job create test_queue
}
ns_job queue test_queue {foo XYZ}

ns_job wait test_queue job0

Collapse
8: Re: ns_job valid procs (response to 7)
Posted by Jonathan Lee on

Ah thank you! Double curly braces allowed the proc to be defined.

As far as adding the proc to a queue, I discovered that the variable could be evaluated if I synced it across interpreters:

ns_eval -sync set globalBar "world"
ns_job queue test_queue {foo $globalBar}

Interestingly, this was only required in the sandbox area I was given to test the functionality (located in a web directory). I'm not sure if it's some configuration in place or this was a quirk inherent to tcl running in {base_dir}/www, but I didn't need to use ns_eval -sync to set the variable anymore once moving the code to the {base_dir}/tcl directory.

Collapse
9: Re: ns_job valid procs (response to 8)
Posted by Brian Fenton on
Happy to hear it worked. Regarding moving to Naviserver, I have migrated 2 applications from AOLserver to Naviserver, and while it's not super challenging, it does take time to get it correct, depending on the size of your codebase. And of course, you will also need to consider upgrading your OS, Tcl version, and database client, among other things. It was definitely worth it for us though.

Brian

Collapse
10: Re: ns_job valid procs (response to 9)
Posted by Jonathan Lee on

Good advice. The time investment is definitely a concern.

Just to document for anyone else who runs into the same mental block I did, the best way to pass args into ns_job is using subst:

set bar "world"
ns_job queue test_queue [subst {foo $bar}]
Collapse
11: Re: ns_job valid procs (response to 10)
Posted by Gustaf Neumann on
the best way to pass args into ns_job is using subst:

Avoid using subst the way you are suggesting, unless you are absolutely certain of the contents of the variable (e.g., bar). Using subst without full knowledge of the variable's content can lead to unexpected issues, no matter you’re using AOLserver or NaviServer.

Concerning a move to NaviServer: the first challenge when moving to NaviServer is adapting your configuration file. To save time, I recommend taking the latest sample configuration file for OpenACS provided in the NaviServer distribution and then updating the necessary values (such as ports, database names, etc.).

consider upgrading your OS, Tcl version, and database client, ...
If your site is subject to external audits, upgrading your operating system, Tcl version, and database client is not optional. Auditors often highlight the risks associated with using outdated or unmaintained software. Moreover, newer versions typically come with essential security improvements that help safeguard your system.