.. |br| raw:: html
********** Session 03 ********** .. figure:: /_static/gateway.jpg :align: center :width: 50% The Wandering Angel http://www.flickr.com/photos/wandering_angel/1467802750/ - CC-BY CGI, WSGI and Living Online =========================== Wherein we discover the gateways to dynamic processes on a server. But First --------- .. rst-class:: large centered Homework Review and Questions Previously ---------- .. rst-class:: build * You've learned about passing messages back and forth with sockets * You've created a simple HTTP server using sockets * You may even have made your server *dynamic* by returning the output of a python script. .. rst-class:: build .. container:: What if you want to pass information to that script? How can you give the script access to information about the HTTP request itself? Stepping Away: The Environment ------------------------------ A computer has an *environment*: .. rst-class:: build .. container:: in \*nix, you can see this in a shell: .. code-block:: bash $ printenv TERM_PROGRAM=iTerm.app ... or in Windows at the command prompt: .. code-block:: posh C:\> set ALLUSERSPROFILE=C:\ProgramData ... or in PowerShell: .. code-block:: posh PS C:\> Get-ChildItem Env: ALLUSERSPROFILE C:\ProgramData ... .. nextslide:: Setting The Environment .. rst-class:: build .. container:: In a ``bash`` shell we can do this: .. code-block:: bash $ export VARIABLE='some value' $ echo $VARIABLE some value or at a Windows command prompt: .. code-block:: posh C:\Users\Administrator\> set VARIABLE='some value' C:\Users\Administrator\> echo %VARIABLE% 'some value' or in PowerShell: .. code-block:: posh PS C:\> $env:VARIABLE = "some value" PS C:\> Get-ChildItem Env:VARIABLE 'some value' .. nextslide:: Viewing the Results These new values are now part of the *environment* .. rst-class:: build .. container:: \*nix: .. code-block:: bash $ printenv ... VARIABLE=some value Windows: .. code-block:: posh C:\> set ... VARIABLE='some value' PowerShell: .. code-block:: posh PS C:\> Get-ChildItem Env: ... VARIABLE 'some value' .. nextslide:: Environment in Python We can see this *environment* in Python, too:: $ python .. code-block:: pycon >>> import os >>> print(os.environ['VARIABLE']) some_value >>> print(os.environ.keys()) ['VERSIONER_PYTHON_PREFER_32_BIT', 'VARIABLE', 'LOGNAME', 'USER', 'PATH', ...] .. nextslide:: Altering the Environment You can alter os environment values while in Python: .. code-block:: pycon >>> os.environ['VARIABLE'] = 'new_value' >>> print(os.environ['VARIABLE']) new_value .. rst-class:: build .. container:: But that doesn't change the original value, *outside* Python: .. code-block:: bash >>> ^D $ echo this is the value: $VARIABLE this is the value: some_value C:\> \Users\Administrator\> echo %VARIABLE% 'some value' .. nextslide:: Lessons Learned .. rst-class:: build .. container:: .. rst-class:: build * Subprocesses inherit their environment from their Parent * Parents do not see changes to environment in subprocesses * In Python, you can actually set the environment for a subprocess explicitly .. code-block:: python subprocess.Popen(args, bufsize=0, executable=None, stdin=None, stdout=None, stderr=None, preexec_fn=None, close_fds=False, shell=False, cwd=None, env=None, # <------- universal_newlines=False, startupinfo=None, creationflags=0) CGI - The Web Environment ========================= .. rst-class:: large centered CGI is little more than a set of standard environmental variables What is CGI ----------- First discussed in 1993, formalized in 1997, the current version (1.1) has been in place since 2004. From the preamble:: This memo provides information for the Internet community. It does not specify an Internet standard of any kind. -- RFC 3875 - CGI Version 1.1: http://tools.ietf.org/html/rfc3875 .. nextslide:: Meta-Variables :: 4. The CGI Request . . . . . . . . . . . . . . . . . . . . . . . 10 4.1. Request Meta-Variables . . . . . . . . . . . . . . . . . 10 4.1.1. AUTH_TYPE. . . . . . . . . . . . . . . . . . . . 11 4.1.2. CONTENT_LENGTH . . . . . . . . . . . . . . . . . 12 4.1.3. CONTENT_TYPE . . . . . . . . . . . . . . . . . . 12 4.1.4. GATEWAY_INTERFACE. . . . . . . . . . . . . . . . 13 4.1.5. PATH_INFO. . . . . . . . . . . . . . . . . . . . 13 4.1.6. PATH_TRANSLATED. . . . . . . . . . . . . . . . . 14 4.1.7. QUERY_STRING . . . . . . . . . . . . . . . . . . 15 4.1.8. REMOTE_ADDR. . . . . . . . . . . . . . . . . . . 15 4.1.9. REMOTE_HOST. . . . . . . . . . . . . . . . . . . 16 4.1.10. REMOTE_IDENT . . . . . . . . . . . . . . . . . . 16 4.1.11. REMOTE_USER. . . . . . . . . . . . . . . . . . . 16 4.1.12. REQUEST_METHOD . . . . . . . . . . . . . . . . . 17 4.1.13. SCRIPT_NAME. . . . . . . . . . . . . . . . . . . 17 4.1.14. SERVER_NAME. . . . . . . . . . . . . . . . . . . 17 4.1.15. SERVER_PORT. . . . . . . . . . . . . . . . . . . 18 4.1.16. SERVER_PROTOCOL. . . . . . . . . . . . . . . . . 18 4.1.17. SERVER_SOFTWARE. . . . . . . . . . . . . . . . . 19 Running CGI ----------- You have a couple of options: .. rst-class:: build .. container:: .. rst-class:: build * Python Standard Library CGIHTTPServer * Apache * IIS (on Windows) * Some other HTTP server that implements CGI (lighttpd, ...?) Let's keep it simple by using the Python module .. nextslide:: Preparations In the class resources for this session, you'll find a directory named ``cgi``. .. rst-class:: build .. container:: Make a copy of that folder in your class working directory. Windows Users, you may have to edit the first line of ``cgi/cgi-bin/cgi_1.py`` to point to your python executable. .. rst-class:: build * Open *two* terminal windows in this ``cgi`` directory * In the first terminal, run ``python -m http.server --cgi`` * Open a web browser and load ``http://localhost:8000/`` * Click on *CGI Test 1* .. nextslide:: Did that work? .. rst-class:: build * Your browser might show a 404 or 403 error * If you see something like that, check the permissions for ``cgi-bin`` *and* ``cgi_1.py`` * The file must be executable, the ``cgi-bin`` directory needs to be readable *and* executable. .. rst-class:: build .. container:: Remember that you can use the bash ``chmod`` command to change permissions in \*nix: ``chmod a+x cgi-bin/cgi_1.py`` Windows users, use the 'properties' context menu to get to permissions, just grant 'full' .. nextslide:: Break It Problems with permissions can lead to failure. So can scripting errors .. rst-class:: build .. container:: .. rst-class:: build * Open ``cgi/cgi-bin/cgi_1.py`` in an editor * Before where it says ``cgi.test()``, add a single line: .. code-block:: python 1 / 0 Reload your browser, what happens now? .. nextslide:: Errors in CGI CGI is famously difficult to debug. There are reasons for this: .. rst-class:: build * CGI is designed to provide access to runnable processes to *the internet* * The internet is a wretched hive of scum and villainy * Revealing error conditions can expose data that could be exploited .. nextslide:: Viewing Errors in Python CGI Back in your editor, add the following lines, just below ``import cgi``: .. rst-class:: build .. container:: .. code-block:: python import cgitb cgitb.enable() Now, reload again. .. nextslide:: cgitb Output .. figure:: /_static/cgitb_output.png :align: center :width: 100% .. nextslide:: Repair the Error Let's fix the error from our traceback. Edit your ``cgi_1.py`` file to match: .. code-block:: python #!/usr/bin/env python import cgi import cgitb cgitb.enable() cgi.test() .. rst-class:: build .. container:: Notice the first line of that script: ``#!/usr/bin/env python``. This is called a *shebang* (short for hash-bang) It tells the system what executable program to use when running the script. CGI Process Execution --------------------- Servers like ``http.server --cgi`` run CGI scripts as a system user called ``nobody``. .. rst-class:: build .. container:: This is just like you calling:: $ ./cgi_bin/cgi_1.py In fact try that now in your second terminal (use the real path), what do you get? Windows folks, you may need ``C:\>python cgi-bin/cgi_1.py`` Notice what is missing? .. nextslide:: There are a couple of important facts about CGI that derive from this: .. rst-class:: build * The script **must** include a *shebang* so that the system knows how to run it. * The script **must** be executable. * The *executable* named in the *shebang* will be called as the *nobody* user. * This is a security feature to prevent CGI scripts from running as a user with any privileges. * This means that the *executable* from the script *shebang* must be one that *anyone* can run. .. nextslide:: The CGI Environment CGI is largely a set of agreed-upon environmental variables. .. rst-class:: build .. container:: We've seen how environmental variables are found in python in ``os.environ`` We've also seen that at least some of the variables in CGI are **not** part of the system environment. Where do they come from? .. nextslide:: CGI Servers Let's find 'em. In a terminal fire up python: .. rst-class:: build .. container:: .. code-block:: ipython In [1]: from http import server In [2]: server.__file__ Out[2]: '/Users/cewing/pythons/parts/opt/lib/python3.5/http/server.py' In [3]: !subl '/Users/cewing/pythons/parts/opt/lib/python3.5/http/server.py' If you don't have the ``subl`` command, or another one that starts your editor, copy this path and open it in your text editor. .. nextslide:: Environmental Set Up From ``http/server.py``, in the ``CGIHTTPRequestHandler`` class, in the ``run_cgi`` method: .. rst-class:: tiny .. code-block:: python env = copy.deepcopy(os.environ) env['SERVER_SOFTWARE'] = self.version_string() env['SERVER_NAME'] = self.server.server_name env['GATEWAY_INTERFACE'] = 'CGI/1.1' ... if self.have_fork: # Unix -- fork as we should ... pid = os.fork() ... try: ... os.execve(scriptfile, args, env) ... else: # Non-Unix -- use subprocess import subprocess ... p = subprocess.Popen(cmdline, ... env = env ) ... .. nextslide:: CGI Scripts And that's it, the big secret. The server takes care of setting up the environment so it has what is needed. .. rst-class:: build .. container:: Now, in reverse. How does the information that a script creates end up in your browser? A CGI Script must print its results to stdout. Use the same method as above to import and open the source file for the ``cgi`` module. Note what ``test`` does for an example of this. .. rst-class:: tiny .. code-block:: python def test(environ=os.environ): ... print("Content-type: text/html") print() try: form = FieldStorage() # Replace with other classes to test those print_directory() print_arguments() print_form(form) ... except: print_exception() .. nextslide:: Recap What the Server Does: .. rst-class:: build * parses the request * sets up the environment, including HTTP and SERVER variables * sends a ``HTTP/1.1 200 OK\r\n`` first line to the client * figures out if the URI points to a CGI script and runs it * appends what comes from the script on stdout and sends that back What the Script Does: .. rst-class:: build * names appropriate *executable* in the *shebang* line * uses os.environ to read information from the HTTP request * builds *any and all* extra **HTTP Headers** |br| (Content-type:, Content-length:, ...) * prints the headers, empty line and script output (body) to stdout In-Class Exercise I ------------------- You've seen the output from the ``cgi.test()`` method from the ``cgi`` module. Let's make our own version of this. .. rst-class:: build .. container:: .. rst-class:: build * In the directory ``cgi-bin`` you will find the file ``cgi_2.py``. * Open that file in your editor. * The script contains some html with text containing placeholders. * You should use Python and the CGI environment to fill the the blanks. * You can view the results of your work by loading ``http://localhost:8000/`` and clicking on *Exercise One* **GO** Getting Data from Users ----------------------- All this is well and good, but where's the *dynamic* stuff? .. rst-class:: build .. container:: It'd be nice if a user could pass form data to our script for it to use. In HTTP, data is often passed to the server as a part of a URL called the *query string* The URL query string is formatted as ``name=value`` pairs, separated by the ampersand (``&``) character The entire query string is separated from other parts of the URL by a question mark:: http://localhost:8000/cgi_bin/somescript.py?a=23&b=46&b=92 .. nextslide:: The Query String in CGI In the ``cgi`` module, we get access to the query string with the ``FieldStorage`` class: .. code-block:: python import cgi form = cgi.FieldStorage() stringval = form.getvalue('a', None) listval = form.getlist('b') .. rst-class:: build * The values in the ``FieldStorage`` are *always* strings * ``getvalue`` allows you to return a default, in case the field isn't present * ``getlist`` always returns a list: empty, one-valued, or as many values as are present In-Class Exercise II -------------------- Let's create a dynamic adding machine. .. rst-class:: build * In the ``cgi-bin`` directory you'll find ``cgi_sums.py``. * In the ``index.html`` file in the ``cgi`` directory, the third link leads to this file. * You will use the structure of that link, and what you learned just now about ``cgi.FieldStorage``. * Complete the cgi script in ``cgi_sums.py`` so that the result of adding all operands sent via the url query is returned. * Return the results as plain text, with the appropriate ``Content-Type`` header. .. nextslide:: My Solution .. rst-class:: build .. code-block:: python form = cgi.FieldStorage() operands = form.getlist('operand') msg = "your total is {total}" try: total = sum(map(int, operands)) msg = msg.format(total=total) except (ValueError, TypeError): msg = "Unable to calculate a sum, please provide integer operands" print("Content-Type: text/plain") print("Content-Length: %s" % len(msg)) print() print(msg) .. nextslide:: Break Time .. rst-class:: centered Let's take a break here, before continuing WSGI ==== .. rst-class:: center large The Web Server Gateway Interface CGI Problems ------------ CGI is great, but there are problems: .. rst-class:: build .. container:: .. rst-class:: build * Code is executed *in a new process* * **Every** call to a CGI script starts a new process on the server * Starting a new process is expensive in terms of server resources * *Especially for interpreted languages like Python* How do we overcome this problem? .. nextslide:: Alternatives to CGI The most popular approach is to have a long-running process *inside* the server that handles CGI scripts. .. rst-class:: build .. container:: FastCGI and SCGI are existing implementations of CGI in this fashion. The PHP scripting language works in much the same way. The Apache module **mod_python** offers a similar capability for Python code. .. rst-class:: build * Each of these options has a specific API * None are compatible with each-other * Code written for one is **not portable** to another This makes it much more difficult to *share resources* A Solution ---------- Enter WSGI, the Web Server Gateway Interface. .. rst-class:: build .. container:: Other alternatives are specific implementations of the CGI standard. WSGI is itself a new standard, not an implementation. WSGI is generalized to describe a set of interactions. Developers can write WSGI-capable apps and deploy them on any WSGI server. Read the original WSGI spec: http://www.python.org/dev/peps/pep-0333 There is also an update for Python 3: |br| https://www.python.org/dev/peps/pep-3333 Apps and Servers ---------------- WSGI consists of two parts, a *server* and an *application*. .. rst-class:: build .. container:: .. container:: A WSGI Server must: .. rst-class:: build * set up an environment, much like the one in CGI * provide a method ``start_response(status, headers, exc_info=None)`` * build a response body by calling an *application*, passing ``environment`` and ``start_response`` as args * return a response with the status, headers and body .. container:: A WSGI Appliction must: .. rst-class:: build * Be a callable (function, method, class) * Take an environment and a ``start_response`` callable as arguments * Call the ``start_response`` method. * Return an *iterable* of 0 or more strings, which are treated as the body of the response. .. nextslide:: Simplified WSGI Server .. code-block:: python from some_application import simple_app def build_env(request): # put together some environment info from the reqeuest return env def handle_request(request, app): environ = build_env(request) iterable = app(environ, start_response) for data in iterable: # send data to client here def start_response(status, headers): # start an HTTP response, sending status and headers # listen for HTTP requests and pass on to handle_request() serve(simple_app) .. nextslide:: Simple WSGI Application Where the simplified server above is **not** functional, this *is* a complete app: .. code-block:: python def application(environ, start_response) status = "200 OK" body = "Hello World\n" response_headers = [('Content-type', 'text/plain'), ('Content-length', len(body))] start_response(status, response_headers) return [body] .. nextslide:: WSGI Middleware A third part of the puzzle is something called WSGI *middleware* .. rst-class:: build .. container:: .. rst-class:: build * Middleware implements both the *server* and *application* interfaces * Middleware acts as a server when viewed from an application * Middleware acts as an application when viewed from a server .. figure:: /_static/wsgi_middleware_onion.png :align: center :width: 38% .. nextslide:: WSGI Data Flow .. rst-class:: build .. container:: .. container:: WSGI Servers: .. rst-class:: large centered **HTTP <---> WSGI** .. container:: WSGI Applications: .. rst-class:: large centered **WSGI <---> app code** .. nextslide:: The WSGI Stack The WSGI *Stack* can thus be expressed like so: .. rst-class:: build large centered **HTTP <---> WSGI <---> app code** .. nextslide:: Using wsgiref The Python standard lib provides a reference implementation of WSGI: .. figure:: /_static/wsgiref_flow.png :align: center :width: 80% .. nextslide:: Apache mod_wsgi You can also deploy with Apache as your HTTP server, using **mod_wsgi**: .. figure:: /_static/mod_wsgi_flow.png :align: center :width: 80% .. nextslide:: Proxied WSGI Servers Finally, it is also common to see WSGI apps deployed via a proxied WSGI server: .. figure:: /_static/proxy_wsgi.png :align: center :width: 80% The WSGI Environment -------------------- REQUEST_METHOD: The HTTP request method, such as "GET" or "POST". This cannot ever be an empty string, and so is always required. SCRIPT_NAME: The initial portion of the request URL's "path" that corresponds to the application object, so that the application knows its virtual "location". This may be an empty string, if the application corresponds to the "root" of the server. PATH_INFO: The remainder of the request URL's "path", designating the virtual "location" of the request's target within the application. This may be an empty string, if the request URL targets the application root and does not have a trailing slash. QUERY_STRING: The portion of the request URL that follows the "?", if any. May be empty or absent. CONTENT_TYPE: The contents of any Content-Type fields in the HTTP request. May be empty or absent. .. nextslide:: The WSGI Environment CONTENT_LENGTH: The contents of any Content-Length fields in the HTTP request. May be empty or absent. SERVER_NAME, SERVER_PORT: When combined with SCRIPT_NAME and PATH_INFO, these variables can be used to complete the URL. Note, however, that HTTP_HOST, if present, should be used in preference to SERVER_NAME for reconstructing the request URL. See the URL Reconstruction section below for more detail. SERVER_NAME and SERVER_PORT can never be empty strings, and so are always required. SERVER_PROTOCOL: The version of the protocol the client used to send the request. Typically this will be something like "HTTP/1.0" or "HTTP/1.1" and may be used by the application to determine how to treat any HTTP request headers. (This variable should probably be called REQUEST_PROTOCOL, since it denotes the protocol used in the request, and is not necessarily the protocol that will be used in the server's response. However, for compatibility with CGI we have to keep the existing name.) .. nextslide:: The WSGI Environment HTTP\_ Variables: Variables corresponding to the client-supplied HTTP request headers (i.e., variables whose names begin with "HTTP\_"). The presence or absence of these variables should correspond with the presence or absence of the appropriate HTTP header in the request. .. rst-class:: build large centered **Seem Familiar?** In-Class Exercise III --------------------- Let's start simply. We'll begin by repeating our first CGI exercise in WSGI .. rst-class:: build * Find the ``wsgi`` directory in the class resources. Copy it to your working directory. * Open the file ``wsgi_1.py`` in your text editor. * We will fill in the missing values using Python and the wsgi ``environ``, just as we use ``os.environ`` in cgi .. rst-class:: build centered **But First** .. nextslide:: Orientation .. code-block:: python if __name__ == '__main__': from wsgiref.simple_server import make_server srv = make_server('localhost', 8080, application) srv.serve_forever() .. rst-class:: build .. container:: Note that we pass our ``application`` function to the server factory We don't have to write a server, ``wsgiref`` does that for us. In fact, you should *never* have to write a WSGI server. .. nextslide:: Orientation .. code-block:: python def application(environ, start_response): response_body = body % ( environ.get('SERVER_NAME', 'Unset'), # server name ... ) status = '200 OK' response_headers = [('Content-Type', 'text/html'), ('Content-Length', str(len(response_body)))] start_response(status, response_headers) return [response_body.encode('utf8')] .. rst-class:: build .. container:: We do not define ``start_response``, the application does that. We *are* responsible for determining the HTTP status. And the content we hand back *must* be ``bytes``, not unicode. .. nextslide:: Running a WSGI Script You can run this script with python:: $ python wsgi_1.py .. rst-class:: build .. container:: This will start a wsgi server. What host and port will it use? Point your browser at ``http://localhost:8080/``. Did it work? Go ahead and fill in the missing bits. Use the ``environ`` passed into ``application`` .. nextslide:: Some Tips WSGI is a long-running process. .. rst-class:: build .. container:: The file you are editing is *not* reloaded after you edit it. You'll need to quit and re-run the script between edits. Notice the use of ``pprint.pprint``, check your terminal for useful output. A WSGI Application ------------------ So now we've learned a bit about the WSGI specification and how a WSGI application can get data that comes in via an HTTP request. .. rst-class:: build .. container:: Let's create a multi-page wsgi application. It will serve a small database of python books. The database (with a very simple api) can be found in ``wsgi/bookdb.py`` .. rst-class:: build * We'll need a listing page that shows the titles of all the books * Each title will link to a details page for that book * The details page for each book will display all the information and have a link back to the list .. nextslide:: Some Questions to Ponder When viewing our first wsgi app, do we see the name of the wsgi application script anywhere in the URL? .. rst-class:: build .. container:: In our wsgi application script, how many applications did we actually have? How are we going to serve different types of information out of a single application? .. nextslide:: Dispatch We have to write an app that will map our incoming request path to some code that can handle that request. .. rst-class:: build .. container:: This process is called ``dispatch``. There are many possible approaches. Let's begin by designing this piece of our app. Open ``bookapp.py`` from the ``wsgi`` folder. We'll do our work here. .. nextslide:: PATH The wsgi environment gives us access to *PATH_INFO*. .. rst-class:: build .. container:: This value is the URI from the client's HTTP request. We can design the URLs that our app will use to assist us in routing. Let's declare that any request for ``/`` will map to the list page. .. container:: We can also say that the URL for a book will look like this:: http://localhost:8080/book/ Writing ``resolve_path`` ------------------------ Let's write a function, called ``resolve_path`` in our application file. .. rst-class:: build * It should take the *PATH_INFO* value from environ as an argument. * It should return the function that will be called. * It should also return any arguments needed to call that function. * This implies of course that the arguments should be part of the PATH .. nextslide:: My Solution .. rst-class:: build .. code-block:: python def resolve_path(path): urls = [(r'^$', books), (r'^book/(id[\d]+)$', book)] matchpath = path.lstrip('/') for regexp, func in urls: match = re.match(regexp, matchpath) if match is None: continue args = match.groups([]) return func, args # we get here if no url matches raise NameError .. nextslide:: Application Updates We need to hook our new dispatch function into the application. .. rst-class:: build * The path should be extracted from ``environ``. * The dispatch function should be used to get a function and arguments * The body to return should come from calling that function with those arguments * If an error is raised by calling the function, an appropriate response should be returned * If the router raises a NameError, the application should return a 404 response .. nextslide:: My Solution .. rst-class:: build .. code-block:: python def application(environ, start_response): headers = [("Content-type", "text/html")] try: path = environ.get('PATH_INFO', None) if path is None: raise NameError func, args = resolve_path(path) body = func(*args) status = "200 OK" except NameError: status = "404 Not Found" body = "

Not Found

" except Exception: status = "500 Internal Server Error" body = "

Internal Server Error

" finally: headers.append(('Content-length', str(len(body)))) start_response(status, headers) return [body.encode('utf8')] Test Your Work -------------- Once you've got your script settled, run it:: $ python bookapp.py .. rst-class:: build .. container:: Then point your browser at ``http://localhost:8080/`` .. rst-class:: build * ``http://localhost/book/id3`` * ``http://localhost/book/id73/`` * ``http://localhost/sponge/damp`` Did that all work as you would have expected? Building the Book List ---------------------- The function ``books`` should return an html list of book titles where each title is a link to the detail page for that book .. rst-class:: build * You'll need all the ids and titles from the book database. * You'll need to build a list in HTML using this information * Each list item should have the book title as a link * The href for the link should be of the form ``/book/`` .. nextslide:: My Solution .. rst-class:: build .. code-block:: python def books(): all_books = DB.titles() body = ['

My Bookshelf

', '
    '] item_template = '
  • {title}
  • ' for book in all_books: body.append(item_template.format(**book)) body.append('
') return '\n'.join(body) Test Your Work -------------- Quit and then restart your application script:: $ python bookapp.py .. rst-class:: build .. container:: .. container:: Then reload the root of your application:: http://localhost:8080/ You should see a nice list of the books in the database. Do you? Click on a link to view the detail page. Does it load without error? Showing Details --------------- The next step of course is to polish up those detail pages. .. rst-class:: build .. container:: .. rst-class:: build * You'll need to retrieve a single book from the database * You'll need to format the details about that book and return them as HTML * You'll need to guard against ids that do not map to books In this last case, what's the right HTTP response code to send? .. nextslide:: My Solution .. rst-class:: build .. code-block:: python def book(book_id): page = """

{title}

Author{author}
Publisher{publisher}
ISBN{isbn}
Back to the list """ book = DB.title_info(book_id) if book is None: raise NameError return page.format(**book) .. nextslide:: Revel in Your Success Quit and restart your script one more time .. rst-class:: build .. container:: Then poke around at your application and see the good you've made And your application is portable and sharable It should run equally well under any `wsgi server `_ .. nextslide:: A Few Steps Further Next steps for an app like this might be: * Create a shared full page template and incorporate it into your app * Improve the error handling by emitting error codes other than 404 and 500 * Swap out the basic backend here with a different one, maybe a Web Service? * Think about ways to make the application less tightly coupled to the pages it serves Homework ======== .. rst-class:: left .. container:: For your homework this week, you'll be creating a wsgi application of your own. .. rst-class:: build .. container:: You'll create an online calculator that can perform several operations You'll need to support: .. rst-class:: build * Addition * Subtraction * Multiplication * Division .. container:: Your users should be able to send appropriate requests and get back proper responses:: http://localhost:8080/multiply/3/5 => 15 http://localhost:8080/add/23/42 => 65 http://localhost:8080/divide/6/0 => HTTP "400 Bad Request" .. nextslide:: Submitting Your Homework .. rst-class:: left .. container:: To submit your homework: .. rst-class:: build * Create a new github repository. Call it ``wsgi-calc``. * Add a python script to it called ``calculator.py``. * Your script should be runnable using ``$ python calculator.py`` * When the script is running, I should be able to view your application in my browser. * I should be able to see a home page that explains how to perform calculations. .. rst-class:: build .. container:: Your repository should include a README.md file. Include all instructions I need to successfully run and view your script. When you are done, send Maria and I an email with a link to your repository. One Last Task ------------- Next week we will be installing Python packages that are not part of the standard library. .. rst-class:: build .. container:: This is a common occurence in web development. But it can be hazardous. In order to practice safe development I am going to ask you to read and follow through a `brief tutorial`_ I've created on the subject. If you have any trouble, or if things do not work the way they are supposed to, please reach out. We will need this to be working next week. .. _brief tutorial: ../../html/presentations/venv_intro.html Wrap-Up ------- For educational purposes, you might wish to take a look at the source code for the ``wsgiref`` module. It's the canonical example of a simple wsgi server >>> import wsgiref >>> wsgiref.__file__ '/full/path/to/your/copy/of/wsgiref.py' ... .. rst-class:: build centered **See you Next Time**