##// END OF EJS Templates
Add keydown listener to commit changes on <enter>
Add keydown listener to commit changes on <enter>

File last commit:

r17502:ef8766d2
r17954:d741bc77
Show More
Background Jobs.ipynb
405 lines | 16.0 KiB | text/plain | TextLexer
Fernando Perez
Add a notebook illustrating the backgroundjobs library.
r4940 {
Brian Granger
Updating a few example notebooks to v3.
r9201 "metadata": {
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "name": "BackgroundJobs"
Brian Granger
Updating a few example notebooks to v3.
r9201 },
"nbformat": 3,
"nbformat_minor": 0,
"worksheets": [
{
"cells": [
{
"cell_type": "markdown",
"metadata": {},
"source": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "# Simple interactive bacgkround jobs with IPython\n",
"\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "We start by loading the `backgroundjobs` library and defining a few trivial functions to illustrate things with."
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"from IPython.lib import backgroundjobs as bg\n",
"\n",
"import sys\n",
"import time\n",
"\n",
"def sleepfunc(interval=2, *a, **kw):\n",
" args = dict(interval=interval,\n",
" args=a,\n",
" kwargs=kw)\n",
" time.sleep(interval)\n",
" return args\n",
"\n",
"def diefunc(interval=2, *a, **kw):\n",
" time.sleep(interval)\n",
" raise Exception(\"Dead job with interval %s\" % interval)\n",
"\n",
"def printfunc(interval=1, reps=5):\n",
" for n in range(reps):\n",
" time.sleep(interval)\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 " print 'In the background...', n\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 " sys.stdout.flush()\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 " print 'All done!'\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 " sys.stdout.flush()"
],
"language": "python",
"metadata": {},
"outputs": [],
Thomas Kluyver
Rerun BackgroundJobs example NB to save in new format, plus Py 3 fixes
r13995 "prompt_number": 1
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"Now, we can create a job manager (called simply `jobs`) and use it to submit new jobs.\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "\n",
"Run the cell below, it will show when the jobs start. Wait a few seconds until you see the 'all done' completion message:"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"jobs = bg.BackgroundJobManager()\n",
"\n",
"# Start a few jobs, the first one will have ID # 0\n",
"jobs.new(sleepfunc, 4)\n",
"jobs.new(sleepfunc, kw={'reps':2})\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "jobs.new('printfunc(1,3)')"
Brian Granger
Updating a few example notebooks to v3.
r9201 ],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
"Starting job # 0 in a separate thread.\n",
"Starting job # 2 in a separate thread.\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Starting job # 3 in a separate thread.\n"
]
},
{
"output_type": "pyout",
"prompt_number": 10,
"text": [
"<BackgroundJob #3: printfunc(1,3)>"
]
},
{
"output_type": "stream",
"stream": "stdout",
"text": [
"In the background... 0\n"
]
},
{
"output_type": "stream",
"stream": "stdout",
"text": [
"In the background... 1\n"
]
},
{
"output_type": "stream",
"stream": "stdout",
"text": [
"In the background... 2\n"
]
},
{
"output_type": "stream",
"stream": "stdout",
"text": [
"All done!\n"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 10
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"You can check the status of your jobs at any time:"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"jobs.status()"
],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Completed jobs:\n",
"0 : <function sleepfunc at 0x314f848>\n",
"2 : <function sleepfunc at 0x314f848>\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "3 : printfunc(1,3)\n",
"\n"
]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 11
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"For any completed job, you can get its result easily:"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "jobs[0].result"
Brian Granger
Updating a few example notebooks to v3.
r9201 ],
"language": "python",
"metadata": {},
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "outputs": [
{
"output_type": "pyout",
"prompt_number": 12,
"text": [
"{'args': (), 'interval': 4, 'kwargs': {}}"
]
}
],
"prompt_number": 12
},
{
"cell_type": "heading",
"level": 2,
"metadata": {},
"source": [
"Errors and tracebacks"
]
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"The jobs manager tries to help you with debugging:"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"# This makes a couple of jobs which will die. Let's keep a reference to\n",
"# them for easier traceback reporting later\n",
"diejob1 = jobs.new(diefunc, 1)\n",
"diejob2 = jobs.new(diefunc, 2)"
],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
"Starting job # 4 in a separate thread.\n",
"Starting job # 5 in a separate thread.\n"
]
}
],
"prompt_number": 13
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"You can get the traceback of any dead job. Run the line\n",
"below again interactively until it prints a traceback (check the status\n",
"of the job):\n"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"print \"Status of diejob1:\", diejob1.status\n",
"diejob1.traceback() # jobs.traceback(4) would also work here, with the job number"
],
"language": "python",
"metadata": {},
"outputs": [
{
Brian E. Granger
Move.
r16102 "output_type": "stream",
"stream": "stdout",
"text": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Status of diejob1: Dead (Exception), call jobs.traceback() for details\n",
"\u001b[1;31m---------------------------------------------------------------------------\u001b[0m\n",
"\u001b[1;31mException\u001b[0m Traceback (most recent call last)\n",
"\u001b[1;32m/home/fperez/usr/opt/virtualenv/ipython-0.13.2/lib/python2.7/site-packages/IPython/lib/backgroundjobs.pyc\u001b[0m in \u001b[0;36mcall\u001b[1;34m(self)\u001b[0m\n",
"\u001b[0;32m 482\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 483\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mcall\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m--> 484\u001b[1;33m \u001b[1;32mreturn\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0m\n",
"\u001b[1;32m<ipython-input-1-fbbbd0d2a1c3>\u001b[0m in \u001b[0;36mdiefunc\u001b[1;34m(interval, *a, **kw)\u001b[0m\n",
"\u001b[0;32m 13\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mdiefunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m2\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m*\u001b[0m\u001b[0ma\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkw\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 14\u001b[0m \u001b[0mtime\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msleep\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m---> 15\u001b[1;33m \u001b[1;32mraise\u001b[0m \u001b[0mException\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m\"Dead job with interval %s\"\u001b[0m \u001b[1;33m%\u001b[0m \u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0m\u001b[0;32m 16\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 17\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mprintfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m1\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mreps\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m5\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\n",
"\u001b[1;31mException\u001b[0m: Dead job with interval 1\n"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 14
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"This will print all tracebacks for all dead jobs:"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"jobs.traceback()"
],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Traceback for: <BackgroundJob #4: <function diefunc at 0x314f668>>\n",
"\u001b[1;31m---------------------------------------------------------------------------\u001b[0m\n",
"\u001b[1;31mException\u001b[0m Traceback (most recent call last)\n",
"\u001b[1;32m/home/fperez/usr/opt/virtualenv/ipython-0.13.2/lib/python2.7/site-packages/IPython/lib/backgroundjobs.pyc\u001b[0m in \u001b[0;36mcall\u001b[1;34m(self)\u001b[0m\n",
"\u001b[0;32m 482\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 483\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mcall\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m--> 484\u001b[1;33m \u001b[1;32mreturn\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
Thomas Kluyver
Rerun BackgroundJobs example NB to save in new format, plus Py 3 fixes
r13995 "\u001b[0m\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "\u001b[1;32m<ipython-input-1-fbbbd0d2a1c3>\u001b[0m in \u001b[0;36mdiefunc\u001b[1;34m(interval, *a, **kw)\u001b[0m\n",
"\u001b[0;32m 13\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mdiefunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m2\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m*\u001b[0m\u001b[0ma\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkw\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 14\u001b[0m \u001b[0mtime\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msleep\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m---> 15\u001b[1;33m \u001b[1;32mraise\u001b[0m \u001b[0mException\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m\"Dead job with interval %s\"\u001b[0m \u001b[1;33m%\u001b[0m \u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0m\u001b[0;32m 16\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 17\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mprintfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m1\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mreps\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m5\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "\u001b[1;31mException\u001b[0m: Dead job with interval 1\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Traceback for: <BackgroundJob #5: <function diefunc at 0x314f668>>\n",
"\u001b[1;31m---------------------------------------------------------------------------\u001b[0m\n",
"\u001b[1;31mException\u001b[0m Traceback (most recent call last)\n",
"\u001b[1;32m/home/fperez/usr/opt/virtualenv/ipython-0.13.2/lib/python2.7/site-packages/IPython/lib/backgroundjobs.pyc\u001b[0m in \u001b[0;36mcall\u001b[1;34m(self)\u001b[0m\n",
"\u001b[0;32m 482\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 483\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mcall\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m--> 484\u001b[1;33m \u001b[1;32mreturn\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
Thomas Kluyver
Rerun BackgroundJobs example NB to save in new format, plus Py 3 fixes
r13995 "\u001b[0m\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "\u001b[1;32m<ipython-input-1-fbbbd0d2a1c3>\u001b[0m in \u001b[0;36mdiefunc\u001b[1;34m(interval, *a, **kw)\u001b[0m\n",
"\u001b[0;32m 13\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mdiefunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m2\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m*\u001b[0m\u001b[0ma\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkw\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 14\u001b[0m \u001b[0mtime\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msleep\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[1;32m---> 15\u001b[1;33m \u001b[1;32mraise\u001b[0m \u001b[0mException\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m\"Dead job with interval %s\"\u001b[0m \u001b[1;33m%\u001b[0m \u001b[0minterval\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0m\u001b[0;32m 16\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
"\u001b[0;32m 17\u001b[0m \u001b[1;32mdef\u001b[0m \u001b[0mprintfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minterval\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m1\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mreps\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m5\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "\u001b[1;31mException\u001b[0m: Dead job with interval 2\n",
Brian Granger
Updating a few example notebooks to v3.
r9201 "\n"
]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 15
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"The job manager can be flushed of all completed jobs at any time:"
]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"jobs.flush()"
],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
Thomas Kluyver
Rerun BackgroundJobs example NB to save in new format, plus Py 3 fixes
r13995 "Flushing 3 Completed jobs.\n",
"Flushing 2 Dead jobs.\n"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 16
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
"After that, the status is simply empty:"
]
},
{
"cell_type": "code",
"collapsed": true,
"input": [
"jobs.status()"
],
"language": "python",
"metadata": {},
"outputs": [],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 17
Brian Granger
Updating a few example notebooks to v3.
r9201 },
{
"cell_type": "markdown",
"metadata": {},
"source": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Jobs have a `.join` method that lets you wait on their thread for completion:"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
},
{
"cell_type": "code",
"collapsed": false,
"input": [
"j = jobs.new(sleepfunc, 2)\n",
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "j.join?"
Brian Granger
Updating a few example notebooks to v3.
r9201 ],
"language": "python",
"metadata": {},
"outputs": [
{
"output_type": "stream",
"stream": "stdout",
"text": [
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "Starting job # 0 in a separate thread.\n"
Brian Granger
Updating a few example notebooks to v3.
r9201 ]
}
],
Brian E. Granger
Updating BG jobs nb from ipython-in-depth.
r17502 "prompt_number": 18
},
{
"cell_type": "markdown",
"metadata": {},
"source": [
"## Exercise\n",
"\n",
"1. Start a new job that calls `sleepfunc` with a 5-second wait\n",
"2. Print a short message that indicates you are waiting (note: you'll need to flush stdout to see that print output appear).\n",
"3. Wait on the job and then print its result."
]
Brian Granger
Updating a few example notebooks to v3.
r9201 }
],
"metadata": {}
}
]
Fernando Perez
Add a notebook illustrating the backgroundjobs library.
r4940 }