Turbogears - Guía rápida
¿Qué es Web Framework?
Web Application Framework o simplemente Web Framework representa una colección de bibliotecas y módulos, que permite a un desarrollador de aplicaciones web escribir aplicaciones, sin tener que preocuparse por detalles de bajo nivel como protocolos, administración de subprocesos, etc.
¿Qué es TurboGears?
TurboGears es un marco de aplicación web escrito en Python. Creada originalmente por Kevin Dangoor en 2005, su última versión TurboGears (ver 2.3.7) es administrada por un grupo de desarrolladores liderados por Mark Ramm y Florent Aide.
TurboGears sigue el paradigma Modelo-Vista-Controlador al igual que la mayoría de los marcos web modernos como Rails, Django, Struts, etc.
Controlador de vista de modelo
MVC es un patrón de diseño de software para desarrollar aplicaciones web. Un patrón de controlador de vista de modelo se compone de tres partes:
Model - El nivel más bajo del patrón es responsable de mantener los datos.
View - Este es responsable de mostrar todos o una parte de los datos al usuario.
Controller - Código de software que controla las interacciones entre el modelo y la vista.
MVC es popular porque aísla la lógica de la aplicación de la capa de interfaz de usuario y admite la separación de preocupaciones. Aquí, el controlador recibe todas las solicitudes de la aplicación y luego trabaja con el modelo para preparar los datos que necesita la vista. La Vista luego usa los datos preparados por el Controlador para generar una respuesta presentable final. La abstracción MVC se puede representar gráficamente de la siguiente manera:
El modelo
El Modelo se encarga de gestionar los datos de la aplicación. Responde a la solicitud de la vista y también responde a las instrucciones del controlador para actualizarse.
La vista
Una presentación de datos en un formato particular, provocada por la decisión de un controlador de presentar los datos. Son sistemas de plantillas basados en scripts muy fáciles de integrar con la tecnología AJAX.
El controlador
El controlador es responsable de responder a la entrada del usuario y realizar interacciones en los objetos del modelo de datos. El Contralor recibe la entrada, valida la entrada y luego realiza la operación comercial que modifica el estado del modelo de datos.
TurboGears se basa en una serie de bibliotecas y herramientas. Estas herramientas han cambiado entre diferentes versiones de TurboGears. Los componentes de la versión actual (ver 2.3.7) se enumeran a continuación.
SQLAlchemy
Es un kit SQL de código abierto que proporciona mapeo de relación de objetos (ORM) para código Python.
Genshi
Este motor de plantillas se utiliza para construir el front-end de las aplicaciones TG. Un sistema de plantillas web combina una plantilla con una determinada fuente de datos para representar páginas web dinámicas.
ToscaWidgets
Es una biblioteca de widgets para generar formularios HTML con controles del lado del servidor. Tosca también actúa como middleware para conectarse con widgets y kits de herramientas de JavaScript.
Caja de cambios
Proporciona un conjunto de comandos para administrar proyectos y aplicaciones de servidor TurboGears. Las aplicaciones TurboGears se pueden implementar en cualquier servidor web compatible con WSGI.
La Interfaz de puerta de enlace del servidor web (WSGI) se ha adoptado como estándar para el desarrollo de aplicaciones web Python. WSGI es una especificación para la interfaz universal entre el servidor web y las aplicaciones web. El paquete wsgiref es una implementación de referencia de WSGI. Se utiliza para agregar soporte WSGI al marco web de TurboGears web. El módulo simple_server de este paquete implementa un servidor HTTP simple que sirve aplicaciones WSGI. Lo usaremos para probar aplicaciones desarrolladas durante este tutorial.
Requisito previo
Python 2.6 o superior. Las versiones anteriores de TurboGears no eran compatibles con Python 3.X. La última versión afirma que funciona bien en Python 3.X. Sin embargo, la documentación oficial de TurboGears todavía se basa en el entorno Python 2.7.
El siguiente comando installs virtualenv -
pip install virtualenv
Este comando necesita administratorprivilegios. Añadirsudo before pipen Linux / Mac OS. Si está en Windows, inicie sesión como administrador. En Ubuntu virtualenv se puede instalar usando su administrador de paquetes.
Sudo apt-get install virtualenv
Una vez instalado, el nuevo entorno virtual se crea en una carpeta.
mkdir newproj
cd newproj
virtualenv venv
Para activar el entorno correspondiente, en Linux/OS X
venv/bin/activate
en Windows
venv\scripts\activate
Ahora estamos listos para install TurboGearsen este entorno. Una instalación mínima de TurboGears se realiza siguiendo el comando:
pip install TurboGears2
El comando anterior se puede ejecutar directamente sin un entorno virtual para una instalación en todo el sistema.
Para instalar TurboGears junto con las herramientas de desarrollo, use el siguiente comando:
pip install tg.devtools
TurboGears tiene un modo mínimo que permite crear rápidamente aplicaciones de un solo archivo. Se pueden crear rápidamente ejemplos y servicios sencillos con un conjunto mínimo de dependencias.
La clase de aplicación en una aplicación TG se hereda de TGControllerclase. Los métodos de esta clase están disponibles para el acceso de@expose decorador de tgmódulo. En nuestra primera aplicación,index()El método se asigna como raíz de nuestra aplicación. La clase TGController también debe importarse desdetg módulo.
from tg import expose, TGController
class MyController(TGController):
@expose()
def index(self):
return 'Hello World turbogears'
A continuación, establezca la configuración de la aplicación y declare el objeto de la aplicación. AppConfig El constructor de clases aquí toma dos parámetros: el atributo mínimo establecido en verdadero y la clase del controlador.
config = AppConfig(minimal = True, root_controller = RootController())
application = config.make_wsgi_app()
los make_wsgi_app() la función aquí construye el objeto de la aplicación.
Para servir esta aplicación, ahora necesitamos iniciar el servidor HTTP. Como se mencionó anteriormente, usaremossimple_server módulo en wsgirefpaquete para configurarlo e iniciarlo. Este módulo tienemake_server() método que requiere el número de puerto y el objeto de la aplicación como argumentos.
from wsgiref.simple_server import make_server
server = make_server('', 8080, application)
server.serve_forever()
Significa que nuestra aplicación será servida en el puerto número 8080 de localhost.
El siguiente es el código completo de nuestra primera aplicación TurboGears:
app.py
from wsgiref.simple_server import make_server
from tg import expose, TGController, AppConfig
class MyController(TGController):
@expose()
def index(self):
return 'Hello World TurboGears'
config = AppConfig(minimal = True, root_controller = MyController())
application = config.make_wsgi_app()
print "Serving on port 8080..."
server = make_server('', 8080, application)
server.serve_forever()
Ejecute el script anterior desde el shell de Python.
Python app.py
Entrar http://localhost:8080 en la barra de direcciones del navegador para ver el mensaje 'Hello World TurboGears'.
los tg.devtoolsde TurboGears contiene Gearbox. Es un conjunto de comandos útiles para la gestión de proyectos TG más complejos. Los proyectos de pila completa se pueden crear rápidamente con el siguiente comando de Gearbox:
gearbox quickstart HelloWorld
Esto creará un proyecto llamado HelloWorld.
Un proyecto de TurboGears contiene los siguientes directorios:
Config - Donde depende la instalación y configuración del proyecto
Controllers - Todos los controladores del proyecto, la lógica de la aplicación web.
i018n - Archivos de traducción para los idiomas admitidos
Lib - Funciones y clases de Python de utilidad
Model - Modelos de base de datos
Public Static Files - CSS, JavaScript e imágenes
Templates - Plantillas expuestas por nuestros controladores.
Tests - El conjunto de pruebas realizadas.
Websetup - Funciones para ejecutar en la configuración de la aplicación.
Cómo instalar un proyecto
Este proyecto ahora debe instalarse. UNsetup.pyya se proporciona en el directorio base del proyecto. Las dependencias del proyecto se instalan cuando se ejecuta este script.
Python setup.py develop
De forma predeterminada, las siguientes dependencias se instalan en el momento de la configuración del proyecto:
- Beaker
- Genshi
- zope.sqlalchemy
- sqlalchemy
- alembic
- repoze.who
- tw2.forms
- tgext.admin ≥ 0.6.1
- WebHelpers2
- babel
Después de la instalación, comience a servir el proyecto en el servidor de desarrollo emitiendo el siguiente comando en el shell:
Gearbox serve –reload –debug
Siga el comando mencionado anteriormente para servir un proyecto de ejemplo prediseñado. Abiertohttp://localhost:8080en el navegador. Esta aplicación de muestra ya preparada ofrece una breve introducción sobre el propio marco TurboGears.
En este proyecto de Hello, el controlador predeterminado se crea en el directorio de controladores como Hello/hello/controllers.root.py. Nos dejamodify root.py con el siguiente código -
from hello.lib.base import BaseController
from tg import expose, flash
class RootController(BaseController):
movie = MovieController()
@expose()
def index(self):
return "<h1>Hello World</h1>"
@expose()
def _default(self, *args, **kw):
return "This page is not ready"
Una vez que una aplicación de trabajo básica está lista, se pueden agregar más vistas en la clase de controlador. En elMycontroller clase anterior, un nuevo método sayHello()está agregado. los@expose() decorador adjunta /sayHelloURL a él. Esta función está diseñada para aceptar un nombre como parámetro de la URL.
Después de iniciar el servidor a través del comando 'gearbox serve', http://localhost:8080. El mensaje Hello World se mostrará en el navegador, incluso si se ingresan las siguientes URL:
http://localhost:8080/
http://localhost:8080/index
Todas estas URL están asignadas a RootController.index()método. Esta clase también tiene_default()que se invocará siempre que una URL no esté asignada a ninguna función específica. El decorador @expose () asigna la respuesta a la URL a una función.
Es posible enviar un parámetro a una función expuesta desde la URL. La siguiente función lee el parámetro de nombre de la URL.
@expose()
def sayHello(self, name):
return '<h3>Hello %s</h3>' %name
El siguiente resultado se verá en el navegador como respuesta a la URL: http://localhost:8080/?name=MVL
Hello MVL
TurboGears asigna automáticamente parámetros de URL a argumentos de función. Nuestra clase RootController se hereda de BaseController. Esto se define comobase.py en el lib folder de aplicación.
Su código es el siguiente:
from tg import TGController, tmpl_context
from tg import request
__all__ = ['BaseController']
def __call__(self, environ, context):
tmpl_context.identity = request.identity
return TGController.__call__(self, environ, context)
TGController.__call__ envía al método Controller al que se enruta la solicitud.
Un evento aunque el contenido HTML se puede devolver al navegador, para una salida más avanzada, siempre se prefiere el uso del motor de plantilla. En un proyecto de pila completa 'iniciado rápidamente' por gearbox, Genshi está habilitado como el renderizador de plantilla predeterminado. En una aplicación mínima, sin embargo, Genshi (o cualquier otro motor de plantilla, como jinja) debe estar instalado y habilitado. El motor de plantillas de Genshi permite escribir plantillas en xhtml puro y las valida para detectar problemas en el momento de la compilación y evitar la publicación de páginas rotas.
Se hace referencia a las plantillas mediante una notación de puntos. En nuestro proyecto Hello, se proporciona un directorio de plantillas para almacenar páginas web de plantillas. Por lo tantosample.html será referido como hello.templates.sample(extensión no mencionada). TurboGears renderiza esta plantilla a través de un decorador de exposición para vincular el método del controlador contg.render_template() función.
La función de controlador expuesta devuelve un objeto de diccionario de Python. Este objeto de diccionario se pasa a su vez a la plantilla vinculada. Los marcadores de posición en la plantilla están llenos de valores de diccionario.
Para empezar, vamos a mostrar una página web con un script html simple. El controlador expuesto devuelve unnull dictionary object ya que no tenemos la intención de enviar ningún dato para que se analice dentro del script HTML.
Cómo crear un HTML de muestra
Nuestra sample.htmlse da a continuación. Asegúrese de que esté almacenado en el directorio de plantillas del proyecto.
<html>
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h2>Hello, Welcome to TurboGears!.</h2>
</body>
</html>
Añadir sample() funcionar en root.py y exponer sample.html a través de él.
@expose("hello.templates.sample")
def sample(self):
return {}
El siguiente resultado se mostrará en el navegador cuando una URL http://localhost:8080/sample se ingresa después de iniciar el servidor web.
Como se mencionó anteriormente, un objeto de diccionario se envía como una colección de parámetros a una plantilla de Genshi. Esta plantilla contiene 'marcadores de posición', que se llenan dinámicamente con los parámetros recibidos del controlador.
Cambiemos el sample() función para enviar un objeto de diccionario a la plantilla de muestra.
@expose("hello.templates.sample")
def sample(self,name):
mydata = {'person':name}
return mydata
Crear sample.html en la carpeta de plantillastemplates\sample.html)
<html>
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h2>Hello, my name is ${person}!.</h2>
</body>
</html>
En el código HTML anterior, ${person}es el marcador de posición. Entrarhttp://localhost:8080/sample?name=MVLcomo URL en el navegador. Esta URL está asignada asample()método en nuestro controlador raíz. Devuelve un objeto de diccionario. Esto se selecciona mediante la página de plantilla vinculada sample.html en el directorio de plantillas. Entonces, MVL sustituye a $ {person} en la página web.
También es posible acceder a los datos del formulario HTML en una función de controlador. El formulario HTML se utiliza para enviar datos del formulario.
El protocolo Http es la base de la comunicación de datos en la World Wide Web. En este protocolo se definen diferentes métodos de recuperación de datos de URL especificadas. La siguiente tabla resume los diferentes métodos http:
No Señor. | Métodos HTTP y descripción |
---|---|
1 | GET Envía datos sin cifrar al servidor. Método más común. |
2 | HEAD Igual que GET, pero sin cuerpo de respuesta |
3 | POST Se utiliza para enviar datos de formularios HTML al servidor. El servidor no almacena en caché los datos recibidos por el método POST. |
4 | PUT Reemplaza todas las representaciones actuales del recurso de destino con el contenido cargado. |
5 | DELETE Elimina todas las representaciones actuales del recurso de destino proporcionadas por una URL |
Crear un formulario HTML
Creemos un formulario HTML y enviemos los datos del formulario a una URL. Guarde la siguiente secuencia de comandos como login.html
<html>
<body>
<form action = "http://localhost:8080/login" method = "get">
<p>Enter Name:</p>
<p><input type = "text" name = "nm" /></p>
<p><input type = "submit" value = "submit" /></p>
</form>
</body>
</html>
Los datos ingresados en este formulario deben enviarse a ‘/login’ URL. Ahora crea una función de controladorloginpage() y exponga la página html anterior.
@expose("hello.templates.login")
def loginpage(self):
return {}
Para recibir los datos del formulario, proporcione un login()controlador, que tiene atributos de formulario como parámetros. aquí‘nm’ es el nombre del campo de entrada de texto en el formulario de inicio de sesión, el mismo se utiliza como parámetro de la función login ().
@expose("hello.templates.sample")
def login(self, nm):
name = nm
return {'person':name}
Como se puede ver, los datos recibidos del formulario de inicio de sesión se envían a la plantilla sample.html (utilizada anteriormente). Es analizado por unGenshi template engine para generar la siguiente salida:
Método POST
Cuando el formulario HTML utiliza el método POST para enviar datos a la URL en el atributo de acción, los datos del formulario no se exponen en la URL. Los datos codificados se reciben en undictargumento por la función del controlador. Los **kw El siguiente argumento es el objeto de diccionario que contiene los datos.
El formulario HTML contiene dos campos de entrada de texto.
<html>
<body>
<form action = "http://localhost:8080/marks" method = "post">
<p>Marks in Physics:</p>
<p><input type = "text" name = "phy" /></p>
<p>Marks in Maths:</p>
<p><input type = "text" name = "maths" /></p>
<p><input type = "submit" value = "submit" /></p>
</form>
</body>
</html>
los marks() El controlador recibe datos del formulario y los envía a sample.htmlmodelo. Código pararoot.py es como sigue -
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose("hello.templates.marks")
def marksform(self):
return {}
@expose("hello.templates.sample")
def marks(self, **kw):
phy = kw['phy']
maths = kw['maths']
ttl = int(phy)+int(maths)
mydata = {'phy':phy, 'maths':maths, 'total':ttl}
return mydata
Finalmente, la plantilla sample.html es la siguiente:
<html>
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h2>Hello, Welcome to TurboGears!.</h2>
<h3>Marks in Physics: ${phy}.</h3>
<h3>Marks in Maths: ${maths}.</h3> <h3>Total Marks: ${total}</h3>
</body>
</html>
Inicie el servidor (si aún no se está ejecutando)
Gearbox server –reload –debug
Entrar http://localhost::8080/marksform en el navegador
los sample.html renderizará la siguiente salida -
Genshi es un lenguaje de plantillas basado en XML. Esto es similar aKid, que solía ser el motor de plantilla para versiones anteriores de TurboGears. Tanto Genshi como Kid se inspiran en otros lenguajes de plantillas bien conocidos comoHSLT, TAL y PHP.
Una plantilla de Genshi consta de directivas de procesamiento. Estas Directivas son elementos y atributos en una plantilla. Las directivas de Genshi se definen en un espacio de nombreshttp://genshi.edgewall.org/. Por lo tanto, este espacio de nombres debe declararse en el elemento raíz de la plantilla.
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
...
</html>
La declaración anterior significa que el espacio de nombres predeterminado está configurado en XHTML y las directivas Genshi tienen el prefijo 'py'.
Directivas de Genshi
Varias directivas se definen en Genshi. La siguiente lista enumera las directivas de Genshi:
- py:if
- py:choose
- py:for
- py:def
- py:match
- py:with
- py:replace
- py:content
- py:attrs
- py:strip
Secciones condicionales
Genshi proporciona dos directivas para la representación condicional de contenido: py: if y py: choose.
py: si
El contenido del elemento de esta directiva se representará solo si la expresión en if clausese evalúa como verdadero. Suponiendo que los datos en el contexto de la plantilla son{‘foo’:True, ‘bar’:’Hello’}, la siguiente directiva -
<div>
<b py:if = "foo">${bar}</b>
</div>
resultará en
Hello
Esta salida, sin embargo, no se renderizaría si ‘foo’ is set to False.
Esta directiva también se puede utilizar como elemento. En este caso<py:if> debe ser cerrado por correspondiente </py:if>
<div>
<py:if test = "foo">
<b>${bar}</b>
</py:if>
</div>
py: elige
El procesamiento condicional avanzado es posible con el uso de py:choose en combinación con py:when y py:otherwisedirectivas. Esta característica es similar aswitch – case construir en C/C++.
Expresión en py:choose La directiva se verifica con diferentes valores identificados con py:whenSe renderizarán alternativas y contenidos correspondientes. Se puede proporcionar una alternativa predeterminada en forma depy:otherwise directiva.
<div py:choose = "foo”>
<span py:when = "0">0</span>
<span py:when = "1">1</span>
<span py:otherwise = "">2</span>
</div>
El siguiente ejemplo ilustra el uso de py:choose y py:whendirectivas. El formulario HTML envía datos a / marca la URL. losmarks() La función redirige las marcas y los resultados en forma de un objeto de diccionario a total.htmlmodelo. La visualización condicional deresult Pass/Fail se logra usando py:choose y py:when directivas.
Script HTML para ingresar marcas (marks.html) es el siguiente:
<html>
<body>
<form action = "http://localhost:8080/marks" method = "post">
<p>Marks in Physics:</p>
<p><input type = "text" name = "phy" /></p>
<p>Marks in Maths:</p>
<p><input type = "text" name = "maths" /></p>
<p><input type = "submit" value = "submit" /></p>
</form>
</body>
</html>
El código completo de root.pyes como sigue. losmarks() El controlador envía marcas y resultados a total.html plantilla -
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose("hello.templates.marks")
def marksform(self):
return {}
@expose("hello.templates.total")
def marks(self, **kw):
phy = kw['phy']
maths = kw['maths']
ttl = int(phy)+int(maths)
avg = ttl/2
if avg ≥ 50:
mydata = {'phy':phy, 'maths':maths, 'total':ttl, 'result':2}
else:
mydata = {'phy':phy, 'maths':maths, 'total':ttl,'result':1}
return mydata
los total.html en la carpeta de plantillas recibe datos del diccionario y los analiza en salida html condicionalmente de la siguiente manera:
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h2>Hello, Welcome to TurboGears!.</h2>
<h3>Marks in Physics: ${phy}.</h3> <h3>Marks in Maths: ${maths}.</h3>
<h3>Total Marks: ${total}</h3>
<div py:choose = "result">
<span py:when = "1"><h2>Result: Fail</h2></span>
<span py:when = "2"><h2>Result: Pass</h2></span>
</div>
</body>
</html>
Inicie el servidor (si aún no se está ejecutando)
Gearbox server –reload –debug
Entrar http://localhost::8080/marksform en el navegador -
los total.html renderizará la siguiente salida -
py: para
Elemento en py: la directiva for se repite para cada elemento en un iterable, normalmente un objeto de lista de Python. Siitems = [1,2,3] está presente en un contexto de plantilla, se puede iterar siguiendo py: for directive -
<ul>
<li py:for = "item in items">${item}</li>
</ul>
Se renderizará la siguiente salida:
1
2
3
El siguiente ejemplo muestra los datos del formulario HTML representados en la plantilla total.html usando py: la directiva for también se puede usar de la siguiente manera:
<py:for each = "item in items">
<li>${item}</li>
</py:for>
Script de formulario HTML
<html>
<body>
<form action = "http://localhost:8080/loop" method="post">
<p>Marks in Physics:</p>
<p><input type = "text" name = "phy" /></p>
<p>Marks in Chemistry:</p>
<p><input type = "text" name = "che" /></p>
<p>Marks in Maths:</p>
<p><input type = "text" name = "maths" /></p>
<p><input type = "submit" value = "submit" /></p>
</form>
</body>
</html>
los loop() El controlador lee los datos del formulario y los envía a total.template en forma de objeto de lista.
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose("hello.templates.marks")
def marksform(self):
return {}
@expose("hello.templates.temp")
def loop(self, **kw):
phy = kw['phy']
maths = kw['maths']
che = kw['che']
l1 = []
l1.append(phy)
l1.append(che)
l1.append(maths)
return ({'subjects':['physics', 'Chemistry', 'Mathematics'], 'marks':l1})
La plantilla temp.html usa py: for loop para representar el contenido del objeto dict en forma de tabla.
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/" lang = "en">
<body>
<b>Marks Statement</b>
<table border = '1'>
<thead>
<py:for each = "key in subjects"><th>${key}</th></py:for>
</thead>
<tr>
<py:for each = "key in marks"><td>${key}</td></py:for>
</tr>
</table>
</body>
</html>
Inicie el servidor (si aún no se está ejecutando)
gearbox server –reload –debug
Entrar http://localhost::8080/marksform en el navegador.
El siguiente resultado se mostrará en el navegador cuando se envíe el formulario anterior.
py: def
Esta directiva se utiliza para crear una macro. Una macro es un fragmento reutilizable de código de plantilla. Al igual que una función de Python, tiene un nombre y, opcionalmente, puede tener parámetros. La salida de esta macro se puede insertar en cualquier lugar de una plantilla.
La directiva py: def sigue la siguiente sintaxis:
<p py:def = "greeting(name)">
Hello, ${name}!
</p>
Esta macro se puede representar con un valor variable para el parámetro 'nombre'.
${greeting('world')} ${greeting('everybody)}
Esta directiva también se puede usar con otra versión de sintaxis de la siguiente manera:
<py:def function = "greeting(name)">
<p>Hello, ${name}! </p>
</py:def>
En el siguiente ejemplo, macro() controlador en root.py envía un dict objeto con dos claves name1 y name2 a la plantilla macro.html.
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose('hello.templates.macro')
def macro(self):
return {'name1':'TutorialPoint', 'name2':'TurboGears'}
Esta plantilla macro.html contiene la definición de una macro llamada saludo. Se utiliza para generar un mensaje de saludo para los datos recibidos del controlador.
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<body>
<h2>py:def example</h2>
<div>
<div py:def = "greeting(name)">
Hello, Welcome to ${name}!
</div>
<b>
${greeting(name1)} ${greeting(name2)}
</b>
</div>
</body>
</html>
Inicie el servidor usando la caja de cambios
gearbox serve –reload –debug
Invoque el controlador macro () ingresando la siguiente URL en el navegador:
http://localhost:8080/macro
La siguiente salida se mostrará en el navegador:
py: con
Esta directiva le permite asignar expresiones a variables locales. Estas variables locales hacen que la expresión interna sea menos detallada y más eficiente.
Suponiendo que x = 50 se da en datos de contexto para una plantilla, el siguiente será el py: con directiva -
<div>
<span py:with = "y = 50; z = x+y">$x $y $z</span>
</div>
Dará como resultado la siguiente salida:
50 50 100
También está disponible una versión alternativa para py: con directiva -
<div>
<py:with = "y = 50; z = x+y">$x $y $z</py:with>
</div>
En el siguiente ejemplo, el controlador macro () devuelve un objeto dict con teclas de nombre, phy y matemáticas.
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose('hello.templates.macro')
def macro(self):
return {'name':'XYZ', 'phy':60, 'maths':70}
La plantilla macro.html agrega valores de claves phy y matemáticas usando py: with directive.
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<body>
<h2>py:with example</h2>
<h3>Marks Statement for : ${name}!</h3> <b>Phy: $phy Maths: $maths <span py:with = "ttl = phy+maths">Total: $ttl</span>
</b>
</body>
</html>
El navegador mostrará el siguiente resultado en respuesta a la URL http://localhost:8080/macro
Directivas de manipulación de estructuras
los py:attrs La directiva agrega, modifica o elimina atributos del elemento.
<ul>
<li py:attrs = "foo">Bar</li>
</ul>
Si foo = {‘class’:’collapse’} está presente en un contexto de plantilla, que representará el fragmento anterior.
<ul>
<li class = "collapse">Bar</li>
</ul>
los py:content La directiva reemplaza cualquier contenido anidado con el resultado de evaluar la expresión -
<ul>
<li py:content = "bar">Hello</li>
</ul>
Dado bar = 'Bye' en los datos de contexto, esto produciría
<ul>
<li>Bye</li>
</ul>
los py:replace La directiva reemplaza el elemento en sí con el resultado de evaluar la expresión -
<div>
<span py:replace = "bar">Hello</span>
</div>
Dado bar = 'Bye' en los datos de contexto, produciría
<div>
Bye
</div>
El contenido de otro documento XML (especialmente el documento HTML) se puede incluir utilizando etiquetas de inclusión en el documento actual. Para habilitar dicha inclusión, el espacio de nombres XInclude debe declararse en el elemento raíz del documento HTML.
<html xmlns = "http://www.w3.org/1999/xhtml" xmlns:xi = "http://www.w3.org/2001/XInclude >
La declaración anterior especifica que la directiva include contiene ‘xi’prefijo. Para agregar contenido de otra página html en el documento actual, use la directiva xi: include de la siguiente manera:
<xi:include href = "somepage.html" />
En el siguiente ejemplo, root.py contiene el controlador include (), que expone include.html.
from hello.lib.base import BaseController
from tg import expose, request
class RootController(BaseController):
@expose('hello.templates.include')
def include(self):
return {}
HTML de encabezado y pie de página
En include.html, se declara el espacio de nombres de inclusión y se añaden los contenidos de header.html y footer.html. Aquí está el script HTML de templates \ include.html -
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:xi = "http://www.w3.org/2001/XInclude">
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<xi:include href = "heading.html" />
<h2>main content </h2>
<xi:include href = "footer.html" />
</body>
</html>
Aquí está el código templates \ header.html -
<html>
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h1>This is page Header</h1>
</body>
</html>
Las siguientes son las plantillas \ footer.html
<html>
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<h3>This is page footer</h3>
</body>
</html>
Inicie el desarrollo usando una caja de cambios y entre http://localhost:8080/includeen el navegador. La salida renderizada será como se muestra a continuación:
De esta manera se puede lograr la construcción modular de vistas. Si el recurso mencionado en la directiva xi: include no está disponible, se generará un error. En tal caso, se puede cargar un recurso alternativo utilizando xi: fallback.
<xi:include href = “main.html”>
<xi:fallback href = ”default.html”/>
</xi.include>
La inclusión de contenido se puede hacer dinámica como un atributo href que puede contener expresiones.
Agregue el siguiente controlador en root.py.
@expose('hello.templates.ref-include')
def refinclude(self):
return {'pages':['heading','main','footer']}
Guarde el siguiente código como ref-include.html en la carpeta de plantillas.
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
xmlns:xi = "http://www.w3.org/2001/XInclude">
<head>
<title>TurboGears Templating Example</title>
</head>
<body>
<xi:include href = "${name}.html" py:for = "name in pages" />
</body>
</html>
Antes de iniciar el servidor, asegúrese de que la carpeta de plantillas tenga un header.html, main.html y footer.html. Entrarhttp://localhost:8082/refinclude en el navegador para obtener el siguiente resultado
los @expose()decorador por defecto muestra contenido html. Sin embargo, esto se puede configurar enjson content type. TurboGears admite la representación json a través detg.jsonify.JSONEncoder (**kwargs)clase. Para renderizar datos json, simplemente pase json como tipo de contenido para exponer decorador.
@expose('json')
def jsondata(self, **kwargs):
return dict(hello = 'World')
Si se ingresa la URL '/ jsondata' en el navegador, responderá mostrando -
{"hello": "World"}
representación jsonp
jsonp significa json con relleno. Funciona de manera similar a la salida json, excepto por el hecho de que proporciona una respuesta de aplicación / javascript con una llamada a una función de javascript que proporciona todos los valores devueltos por el controlador como argumentos de función.
Para habilitar la representación jsonp, primero debe agregarlo a la lista de motores requeridos dentro de su aplicación: config/app_cfg.py -
base_config.renderers.append('jsonp')
Escriba su decorador de exposición de la siguiente manera:
@expose('json')
@expose('jsonp')
def jsonpdata (self, **kwargs):
return dict(hello = 'World')
Al acceder a / jsonpdata? Callback = callme, debería ver -
callme({"hello": "World"});
A veces, una aplicación web puede requerir una estructura de URL que tenga más de un nivel. TurboGears puede atravesar la jerarquía de objetos para encontrar el método apropiado que pueda manejar su solicitud.
Un proyecto de 'inicio rápido' con caja de cambios tiene una clase BaseController en la carpeta lib del proyecto. Está disponible como 'Hello / hello / lib / base.py'. Sirve como clase base para todos los subcontroladores. Para agregar un subnivel de URL en la aplicación, diseñe una subclase llamada BlogController derivada de BaseController.
Este BlogController tiene dos funciones de controlador, index () y post (). Ambos están diseñados para exponer una plantilla cada uno, blog.html y post.html.
Note - Estas plantillas se colocan dentro de una subcarpeta - templates / blog
class BlogController(BaseController):
@expose('hello.templates.blog.blog')
def index(self):
return {}
@expose('hello.templates.blog.post')
def post(self):
from datetime import date
now = date.today().strftime("%d-%m-%y")
return {'date':now}
Ahora declare un objeto de esta clase en la clase RootController (en root.py) de la siguiente manera:
class RootController(BaseController):
blog = BlogController()
Otras funciones de controlador para URL de nivel superior estarán allí en esta clase como antes.
Cuando una URL http://localhost:8080/blog/se ingresa, se asignará a la función del controlador index () dentro de la clase BlogController. Similar,http://localhost:8080/blog/post invocará la función post ().
El código para blog.html y post.html es el siguiente:
Blog.html
<html>
<body>
<h2>My Blog</h2>
</body>
</html>
post.html
<html>
<body>
<h2>My new post dated $date</h2>
</body>
</html>
Cuando una URL http://localhost:8080/blog/ se ingresa, producirá la siguiente salida:
Cuando una URL http://localhost:8080/blog/post se ingresa, producirá la siguiente salida:
Uno de los aspectos más esenciales de cualquier aplicación web es presentar la interfaz de usuario para un usuario. HTML proporciona una etiqueta <form> que se utiliza para diseñar una interfaz. Los elementos del formulario, como entrada de texto, radio, selección, etc. se pueden utilizar de forma adecuada. Los datos ingresados por el usuario se envían en forma de mensaje de solicitud Http al script del lado del servidor mediante el método GET o POST.
El script del lado del servidor tiene que recrear los elementos del formulario a partir de los datos de la solicitud http. Entonces, en este efecto, los elementos del formulario deben definirse dos veces: una en HTML y nuevamente en el script del lado del servidor.
Otra desventaja del uso de formularios HTML es que es difícil (si no imposible) representar los elementos del formulario de forma dinámica. HTML en sí no proporciona ninguna forma de validar la entrada del usuario.
ToscaWidgets2
TurboGears se basa en ToscaWidgets2, una biblioteca de validación y representación de formularios flexible. Usando ToscaWidgets, podemos definir los campos de formulario en nuestro script Python y representarlos usando una plantilla HTML. También es posible aplicar la validación al campo tw2.
La biblioteca ToscaWidgets es una colección de muchos módulos. Algunos módulos importantes se enumeran a continuación:
tw2.core- Proporciona una funcionalidad básica. Los widgets de este módulo no deben estar disponibles para el usuario final.
tw2.forms- Esta es una biblioteca de formularios básicos. Contiene widgets para campos, conjuntos de campos y formularios.
tw2.dynforms - Contiene la funcionalidad de formularios dinámicos.
tw2.sqla - Esta es una interfaz para la base de datos SQLAlchemy.
tw2.forms
Contiene una clase Form, que actúa como base para formularios personalizados. Existe una clase TableForm que es útil para representar campos en una tabla de dos columnas. ListForm presenta sus campos en una lista desordenada.
No Señor. | Campos y descripción |
---|---|
1 | TextField Un campo de entrada de texto de una sola línea |
2 | TextArea Campo de entrada de texto de varias líneas |
3 | CheckBox Presenta una caja rectangular marcable con etiqueta |
4 | CheckBoxList Casillas de verificación de pf de grupo de selección múltiple |
5 | RadioButton Un botón de alternancia para seleccionar / deseleccionar |
6 | RadioButtonList Grupo de botones de radio mutuamente excluyentes |
7 | PasswordField Similar a Textfield pero las teclas de entrada no se revelan |
8 | CalendarDatePicker Permite al usuario elegir una fecha. |
9 | SubmitButton Botón para enviar un formulario |
10 | ImageButton Botón en el que se puede hacer clic con una imagen en la parte superior |
11 | SingleSelectField Permite la selección de un solo elemento de una lista |
12 | MultipleSelectField Permite la selección de varios elementos de la lista. |
13 | FileField Campo para cargar archivo |
14 | EmailField Un campo de entrada de correo electrónico |
15 | URLField Un campo de entrada para ingresar URL |
dieciséis | NumberField Una caja de números |
17 | RangeField Un control deslizante de números |
En el siguiente ejemplo, se construye un formulario que utiliza algunos de estos widgets. Si bien la mayoría de estos widgets se definen en tw2.forms, CalendarDateField se define en el módulo tw2.Dynforms. Por lo tanto, ambos módulos junto con tw2.core se importan al principio:
import tw2.core as twc
import tw2.forms as twf
import tw2.dynforms as twd
Un formulario ToscaWidgets es una clase derivada de tw2.forms.formclase base. Los widgets necesarios se colocan dentro de un objeto Layout. En este ejemplo,TableLayoutse utiliza. Los widgets se representan en una tabla de dos columnas. La primera columna muestra el título y la segunda columna muestra el campo de entrada o selección.
Se crea un objeto TextField utilizando el siguiente constructor:
twf.TextField(size, value = None)
Si no se menciona, el objeto TextField toma un tamaño predeterminado e inicialmente está en blanco. Al declarar el objeto TextArea, se puede mencionar el número de filas y columnas.
twf.TextArea("",rows = 5, cols = 30)
El objeto NumberField es un TextField que solo puede aceptar dígitos. Las flechas hacia arriba y hacia abajo se generan en el borde derecho para aumentar o disminuir el número dentro de él. El valor inicial también se puede especificar como un argumento en el constructor.
twf.NumberField(value)
Justo a la derecha de un cuadro CalendarDatePicker, se muestra un botón de calendario. Cuando se presiona, aparece un selector de fecha. El usuario puede escribir manualmente una fecha en el cuadro o seleccionar en el selector de fechas.
twd.CalendarDatePicker()
El objeto EmailField presenta un TextField, pero el texto debe estar en formato de correo electrónico.
EmailID = twf.EmailField()
El siguiente formulario también tiene un RadioButtonList. El constructor de esta clase contiene un objeto List como un parámetro de valor de opciones. Se renderizará un botón de opción para cada opción. La selección predeterminada se especifica con el parámetro de valor.
twf.RadioButtonList(options = ["option1","option2"],value = option1)
CheckBoxList presenta casillas de verificación para cada opción de la lista.
twf.CheckBoxList(options = [option1, option2, option3])
La lista desplegable se llama como SingleSelectfield en esta biblioteca de ToscaWidgets. Los elementos de un objeto de lista correspondiente al parámetro de opciones forman la lista desplegable. El título visible se establece como un valor del parámetro prompt_text.
twf.SingleSelectField(prompt_text = 'text', options=['item1', 'item2', 'item3'])
De forma predeterminada, el formulario muestra un botón Enviar con el título "Guardar". Para mostrar otro título, cree un objeto SubmitButton y especifíquelo como parámetro de valor.
twf.SubmitButton(value = 'Submit')
El formulario se envía a una URL, que se especifica como un valor de parámetro de acción del formulario. De forma predeterminada, los datos del formulario se envían mediante el método http POST.
action = 'URL'
En el siguiente código, un formulario denominado AdmissionForm está diseñado utilizando los widgets explicados anteriormente. Agregue este código en root.py antes de la clase RootController.
class AdmissionForm(twf.Form):
class child(twf.TableLayout):
NameOfStudent = twf.TextField(size = 20)
AddressForCorrespondance = twf.TextArea("", rows = 5, cols = 30)
PINCODE = twf.NumberField(value = 431602)
DateOfBirth = twd.CalendarDatePicker()
EmailID = twf.EmailField()
Gender = twf.RadioButtonList(options = ["Male","Female"],value = 'Male')
Subjects = twf.CheckBoxList(options = ['TurboGears', 'Flask', 'Django', 'Pyramid'])
MediumOfInstruction = twf.SingleSelectField(prompt_text = 'choose',
options = ['English', 'Hindi', 'Marathi', 'Telugu'])
action = '/save_form'
submit = twf.SubmitButton(value ='Submit')
Ahora guarde este código a continuación como twform.html en el directorio de plantillas -
<!DOCTYPE html>
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<head>
<title>TurboGears Form Example</title>
</head>
<body>
<div id = "tw form">
${form.display(value = dict(title = 'default title'))}
</div>
</body>
</html>
En la clase RootController (en root.py), agregue la siguiente función de controlador:
@expose('hello.templates.twform')
def twform(self, *args, **kw):
return dict(page = 'twform', form = MovieForm)
En la clase AdmissionForm, tenemos stipulated/save_formcomo URL de acción. Por lo tanto, agreguesave_form() función en RootController.
@expose()
def save_movie(self, **kw):
return str(kw)
Asegúrese de que el servidor esté funcionando (utilizando la caja de cambios). Entrarhttp://localhost:8080/twform en el navegador.
Al presionar el botón enviar, se publicarán estos datos en save_form() URL, que mostrará los datos del formulario en forma de un objeto de diccionario.
{
'EmailID': u'[email protected]',
'NameOfStudent': u'Malhar Lathkar',
'Gender': u'Male',
'PINCODE': u'431602',
'DateOfBirth': u'2015-12-29',
'Subjects': [u'TurboGears', u'Flask', u'Django'],
'MediumOfInstruction': u'',
'AddressForCorrespondance': u'Shivaji Nagar\r\nNanded\r\nMaharashtra'
}
Una buena biblioteca de widgets de formularios debe tener una función de validación de entrada. Por ejemplo, el usuario debe ser forzado a ingresar datos en un campo obligatorio, o verificar si un campo de correo electrónico contiene un correo electrónico válido, sin recurrir a ningún otro medio programático (como la función de JavaScript) para la validación.
Las primeras versiones de la biblioteca de formularios ToscaWidgets solían depender del módulo FormEncode para el soporte de validación. ToscaWidgets2 ahora tiene soporte de validación incorporado disponible en el módulo tw2.core. Sin embargo, todavía es posible utilizar técnicas de validación de FormEncode.
Para someter un formulario ToscaWidgets a validación, se utiliza @validate decorator.
@validate(form, error_handler, validators)
los ’form’ es el objeto de formulario ToscaWidgets que se va a validar.
los ‘error-handler’ es el método de controlador utilizado para manejar errores de formulario.
los ‘validators’ son un objeto de diccionario que contiene validadores FormEncode.
Tipos de validadores
El módulo tw2.core contiene una clase de validación de la que se heredan otros validadores. También es posible diseñar un validador personalizado basado en él. Algunos de los validadores importantes se describen a continuación:
LengthValidator- Compruebe si un valor tiene una longitud prescrita. Los límites mínimos y máximos se definen con parámetros mínimos y máximos. Los mensajes personalizados para la longitud por debajo y por encima del mínimo y el máximo se pueden especificar como parámetro demasiado corto y largo.
tw2.core.LengthValidator(min = minval, max = maxval,
msgs = { 'tooshort': (‘message for short length’),
'toolong': (‘message for long length)})
RangeValidator- Usualmente usado junto con RangeField. Es útil para validar el valor de un campo numérico dentro de los límites mínimo y máximo. Los mensajes para parámetros demasiado cortos y largos se pueden personalizar.
tw2.core.RangeValidator(min = minval, max = maxval,
msgs = { 'tooshort': (‘message for short length’),
'toolong': (‘message for long length)})
IntValidator- Esta clase se deriva del RangeValidator. Esto se usa normalmente para validar si la entrada en un campo de texto normal contiene datos enteros. Se pueden establecer límites mínimos y máximos, así como mensajes de error. Además, el mensaje de error para la entrada no entera se puede especificar como parámetro 'notint'.
tw2.core.IntValidator(msgs = {‘notint’:’Must be Integer’})
OneOfValidator - Este validador obliga al usuario a seleccionar un valor de las opciones disponibles solo en la lista.
tw2.core.OneOfValidator(values = [option1, option2,..],
msgs = {‘notinlist’:’Not in List’}}
DateValidator- Muy útil para asegurar que la entrada del usuario sea una fecha válida. El formato de fecha (el predeterminado es YMD) y el mensaje de error se pueden personalizar. También se pueden especificar límites de fecha mínimos y máximos. DateTimeValidator también está disponible para verificar el objeto de la clase DateTime.
tw2.core.DateValidator(msgs = {format = ’%Y-%m-%d’,
'baddatetime': ('baddate', ('Must follow date format $format_str'))}
EmailValidator- Valida la entrada del usuario con una dirección de correo electrónico válida. Esta clase se hereda de un RegexValidator más general.
tw2.core.EmailValidator(msgs = {'badregex': ('bademail',
('Must be a valid email address')) }
UrlValidator- Esta clase también se hereda de RegexValidator. Valida la entrada del usuario para una URL válida.
tw2.core.UrlValidator(msgs = {'badregex': ('badurl', ('Must be a valid URL’)) }
MatchValidator- Confirma si el valor de un campo coincide con el otro. Esto es especialmente útil cuando el usuario debe elegir y confirmar un campo de contraseña. El uso típico de MatchValidator se muestra a continuación:
import tw2.core as twc
import tw2.forms as twf
class AdmissionForm(twf.Form):
class child(twf.TableLayout):
validator = twc.MatchValidator('pw', 'pwconfirm')
pw = twf.PasswordField()
pwconfirm = twf.PasswordField()
También es posible construir un validador compuesto, donde se desea que la validación sea exitosa, si cualquiera de las comprobaciones pasa. En otros casos, es posible que desee que la validación tenga éxito, solo si la entrada pasa todas las comprobaciones. Para ello, tw2.core proporciona los validadores Any y All, que son subclases del CompoundValidator extensible.
TurboGears proporciona un sistema de mensajería muy conveniente para notificar información al usuario de una manera no molesta. La clase TGFlash en el módulo tg proporciona soporte para flashear mensajes que se almacenan en una cookie simple. Esta clase admite la obtención de mensajes flash en el lado del servidor y en el lado del cliente a través de JavaScript.
los render()El método de la clase TGFlash, cuando se usa desde Python, puede invocarse desde la plantilla para representar un mensaje flash. Si se usa en JavaScript, proporciona un objeto WebFlash. Exponepayload() y render() métodos para recuperar el mensaje flash actual y representarlo desde JavaScript.
Cuando se crea un proyecto de TurboGears usando 'inicio rápido', tiene una plantilla Master.html. Contiene la definición de una variable de ese objeto flash. El contenido de este mensaje flash recibido del controlador sustituye al marcador de posición marcado en esta plantilla.
<py:with vars = "flash = tg.flash_obj.render('flash', use_js = False)">
<div py:if = "flash" py:replace = "Markup(flash)" />
</py:with>
los tg.flash_obj es el objeto WebFlash, que está disponible dentro de cualquier plantilla renderizada al incluir master.htmlmodelo. Este objeto permite recuperar el mensaje flash actual y mostrarlo.
Los mensajes Flash se almacenan en una cookie (cuyo nombre por defecto es webflash) usando tg.flash()método. A continuación, se le pasan los parámetros de mensaje y estado.
tg.flash('Message', 'status')
Si el método que se llama flash realiza una redirección, entonces el flash será visible dentro de la página redirigida. Si el método expone directamente una plantilla, el flash será visible dentro de la propia plantilla.
La apariencia del mensaje flash se puede personalizar aplicando estilos CSS al código de estado. Un proyecto de 'inicio rápido' contiene códigos de error, advertencia, información y estado correcto personalizados por una hoja de estilo public / css / style.css. También se pueden agregar más códigos de estado con estilos.
#flash > .warning {
color: #c09853;
background-color: #fcf8e3;
border-color: #fbeed5;
}
#flash > .ok {
color: #468847;
background-color: #dff0d8;
border-color: #d6e9c6;
}
#flash > .error {
color: #b94a48;
background-color: #f2dede;
border-color: #eed3d7;
}
#flash > .info {
color: #3a87ad;
background-color: #d9edf7;
border-color: #bce8f1;
}
Esta hoja de estilo externa debe incluirse en la plantilla:
<link rel = "stylesheet" type = "text/css" media = "screen"
href = "${tg.url('/css/style.css')}" />
La configuración de cualquier soporte de mensajes Flash se puede lograr estableciendo parámetros para el método configure () del objeto TGFlash o en app_cfg.py (en la carpeta de configuración). Los parámetros configurables son:
No Señor. | Parámetros y descripción |
---|---|
1 | flash.cookie_name Nombre de la cookie utilizada para almacenar mensajes flash. El valor predeterminado eswebflash. |
2 | flash.default_status Estado del mensaje predeterminado si no se especifica (ok de forma predeterminada) |
3 | flash.template Usado como flash template cuando se renderiza. |
4 | flash.allow_html Vueltas on/off escaping in flash messages, HTML predeterminado no está permitido. |
5 | flash.js_call Código JavaScript que se ejecutará al mostrar el flash de JavaScript. El valor predeterminado eswebflash.render() |
6 | flash.js_template string.Template instancia utilizada para reemplazar el soporte completo de JavaScript para mensajes flash. |
pop_payload() - función fetches current flash message, statuse información relacionada. Recibir el mensaje flash eliminará la cookie.
render(container_id, use_js=True) - Renderice el mensaje flash dentro de la plantilla o proporcione soporte Javascript para ellos.
container_id es el DIV donde se mostrarán los mensajes, mientras que use_js cambia entre renderizar el flash como HTML o para el uso de JavaScript.
status - Obtener solo el estado actual del flash, obtener el estado del flash eliminará la cookie.
message - Obtener solo el mensaje flash actual, al recibir el mensaje flash se eliminará la cookie.
¿Cómo hacer un mensaje flash simple?
En el siguiente ejemplo, se proporciona un método flash () en la clase del controlador raíz. Llama a un mensaje flash () que se representa en la plantilla expuesta, flash.html
from hello.lib.base import BaseController
from tg import expose, flash, redirect, request
class RootController(BaseController):
@expose('hello.templates.flash')
def flash(self, user = None):
if user:
flash(message = "Welcome "+user,status = "ok")
else:
flash(message = "Welcome Guest",status = "info")
return {}
El código para hacer flash.html en la carpeta de plantillas es la siguiente
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
xmlns:xi = "http://www.w3.org/2001/XInclude">
<head>
<title>TurboGears 2.3: Flash messages>/title>
<link rel = "stylesheet" type = "text/css" media = "screen"
href = "${tg.url('/css/style.css')}" />
<py:with vars = "flash = tg.flash_obj.render('flash', use_js = False)">
<div py:if = "flash" py:replace = "Markup(flash)" />
</py:with>
</head>
<body>
<h2>Hello TurboGears</h2>
</body>
</html>
Inicie el servidor e ingrese http://localhost:8080/flash?user=MVL en el navegador
Cambiar URL a http://localhost:8080/flash y vea el mensaje flash con un formato diferente según la definición en style.css
A menudo se requiere mantener datos de navegación simples adjuntos al navegador de un usuario. Las sesiones son la técnica más utilizada. La sesión representa datos que no necesitan almacenarse en una forma más persistente como un archivo de disco o una base de datos.
Sin embargo, los datos de sesión en TurboGears pueden estar respaldados por el sistema de archivos, la base de datos o los valores de cookies con hash. Generalmente, una pequeña cantidad de datos de sesión se guarda en cookies, pero para el mayor volumen de datos de sesión se usa MemCache.
MemCache es un demonio a nivel de sistema. Proporciona un acceso rápido a los datos almacenados en caché y es extremadamente escalable. Sin embargo, está diseñado para su uso únicamente en servidores seguros y, por lo tanto, debe ser mantenido y protegido por sysadmin.
Vaso de precipitados en la gestión de sesiones
TurboGears usa Beaker para la gestión de sesiones. Un proyecto iniciado rápidamente por la caja de cambios está configurado de forma predeterminada para usar cookies hash para almacenar datos de sesión.
Cada vez que un cliente se conecta, el middleware de la sesión (Beaker) inspeccionará la cookie utilizando el nombre de la cookie, que se ha definido en el archivo de configuración. Si no se encuentra la cookie, se instalará en el navegador. En todas las visitas posteriores, el middleware encontrará la cookie y la utilizará.
Para habilitar la gestión de sesiones, la clase de sesión debe incorporarse en el proyecto siguiendo la declaración de importación:
from tg import session
Para guardar los datos en una variable de sesión:
session[‘key’] = value
session.save()
Para recuperar la variable de sesión:
return session[‘key’]
Tenga en cuenta que debe guardar explícitamente la sesión para que sus claves se almacenen en esa sesión.
los delete() El método del objeto de sesión borrará todas las sesiones de usuario -
session.delete()
Aunque no es habitual eliminar todas las sesiones de usuario en un entorno de producción determinado, normalmente lo hará para limpiar después de que se hayan realizado las pruebas funcionales o de usabilidad.
A continuación se muestra un ejemplo sencillo para demostrar sesiones. La clase RootController tiene unsetsession() método que establece una variable de sesión.
from hello.lib.base import BaseController
from tg import expose, session
class RootController(BaseController):
@expose()
def setsession(self):
session['user'] = 'MVL'
session.save()
str = "<b>sessionVariable set to "+session['user']
str = str+"<br><a href = '/getsession'>click here to retrieve</a></b>"
return str
@expose()
def getsession(self):
return "<b>value of session variable retrieved " +session['user'] +"</b>"
Entrar http://localhost:8080/setsession
Un enlace en el navegador conduce a http://localhost:8080/getsession que recupera y muestra la variable de sesión -
Para mejorar el rendimiento de una aplicación web, especialmente si está involucrada en operaciones prolongadas, se utilizan técnicas de almacenamiento en caché. TurboGears proporciona dos tipos de técnicas de almacenamiento en caché:
Whole-page Caching
Funciona a nivel de protocolo HTTP para evitar solicitudes completas al servidor al hacer que el navegador del usuario o un servidor proxy intermedio (como Squid) intercepte la solicitud y devuelva una copia en caché del archivo.
Application-level Caching
Esto funciona dentro del servidor de aplicaciones para almacenar en caché los valores calculados, a menudo los resultados de consultas complejas de la base de datos, para que las solicitudes futuras puedan evitar tener que volver a calcular los valores. Para las aplicaciones web, el almacenamiento en caché a nivel de aplicación proporciona una forma flexible de almacenar en caché los resultados de consultas complejas para que la carga total de un método de controlador dado se pueda reducir a unas pocas consultas específicas del usuario o del caso y la sobrecarga de procesamiento de una plantilla .
Almacenamiento en caché a nivel de aplicación
Como se mencionó anteriormente, el proyecto TurboGears 'de inicio rápido' está configurado para habilitar el paquete Beaker para la compatibilidad con el almacenamiento en caché. Beaker admite los siguientes back-end utilizados para el almacenamiento en caché:
memory- Utilizado para almacenamiento por proceso. Es extremadamente rápido.
filesystem - almacenamiento por proceso y multiproceso.
DBM database - por proceso, multiproceso, bastante rápido.
SQLAlchemy database- almacenamiento por servidor de base de datos. Más lento en comparación con las opciones dadas anteriormente.
Memcached - Caché basado en memoria multiservidor.
Almacenamiento en caché del controlador
Para un almacenamiento en caché rápido del controlador, cached()decorador está disponible. Todo el cuerpo del controlador se almacena en caché en función de varios parámetros de solicitud. La definición detg.decorators.cached() decorador es el siguiente
tg.decorators.cached(key, expire, type,
query-args, cache_headers, invalidate_on_startup, cache_response)
La descripción de los parámetros es la siguiente:
No Señor. | Parámetros y descripción |
---|---|
1 | key Especifica los parámetros del controlador que se utilizan para generar la clave de caché. |
2 | expire Tiempo en segundos antes de que expire la caché, el valor predeterminado es "nunca". |
3 | Type dbm, memoria, archivo, memcached o Ninguno. |
4 | cache_headers Una tupla de nombres de encabezados que indican encabezados de respuesta. |
5 | invalidate_on_startup Si es Verdadero, la caché se invalida cada vez que se inicia o se reinicia la aplicación. |
6 | cache_response la respuesta debe almacenarse en caché o no, el valor predeterminado es True. |
A continuación se muestra un ejemplo de almacenamiento en caché del controlador:
@cached(expire = 100, type = 'memory')
@expose()
def simple(self):
return "This is a cached controller!"
Almacenamiento en caché de nivel de plantilla
El motor de plantillas de Genshi recupera la plantilla de un caché si su contenido no ha cambiado. El tamaño predeterminado de esta caché es 25. De forma predeterminada, la recarga automática de plantillas es verdadera. Para mejorar el rendimiento, se pueden realizar los siguientes ajustes enapp_cfg.py -
[app:main]
genshi.max_cache_size = 100
auto_reload_templates = false
Para almacenar en caché una plantilla, solo tiene que devolver el tg_cache opción del controlador que procesa la plantilla en caché.
El tg_cache es un diccionario que acepta las siguientes claves:
key - La clave de caché. Default: Ninguna.
expire - cuánto tiempo debe permanecer vivo el caché. Default: nunca expira
type - memoria, dbm, memcached. Default: dbm.
El siguiente ejemplo ilustra el almacenamiento en caché de plantillas:
@expose(hello.templates.user')
def user(self, username):
return dict(user = username, tg_cache = dict(key = user, expire = 900))
Aunque es posible usar SQL en la aplicación TurboGears para realizar operaciones CRUD en cualquier base de datos relacional, es recomendable usar SQLAlchemy, un kit de herramientas de Python es un poderoso asignador de relaciones de objetos que brinda a los desarrolladores de aplicaciones todo el poder y la flexibilidad de SQL. Además de la compatibilidad con bases de datos basadas en SQL a través de SQLAlchemy, TurboGears también admite la base de datos MongoDB a través de Ming. En esta sección, se analiza la funcionalidad de SQLAlchemy.
¿Qué es ORM (mapeo relacional de objetos)?
La mayoría de las plataformas de lenguajes de programación están orientadas a objetos. Los datos en los servidores RDBMS, por otro lado, se almacenan como tablas. El mapeo de relaciones de objeto es una técnica de mapeo de parámetros de objeto a la estructura de tabla RDBMS subyacente. Una API ORM proporciona métodos para realizar operaciones CRUD sin tener que escribir sentencias SQL sin procesar.
Cuando se crea un proyecto de TurboGears usando el comando de 'inicio rápido' del kit de herramientas de la caja de engranajes, el soporte de SQLAlchemy está habilitado de manera predeterminada por los siguientes ajustes de configuración:
config['use_sqlalchemy'] = True
config['sqlalchemy.url'] = 'sqlite:///devdata.db'
El proyecto de 'inicio rápido' también crea un paquete de modelos dentro de él. Por ejemplo, un proyecto 'Hello' tendrá modelo Hello \ hello \. Los siguientes archivos se crean en este paquete:
__init__.py- Aquí es donde se configura el acceso a la base de datos. Los objetos del modelo de la aplicación se importan en este módulo. También tiene una DBSession - un administrador de sesión global y también una DeclarativeBase, que es una clase base para todas las clases modelo.
auth.py- Aquí es donde se definen los modelos utilizados por la pila de autenticación. Los modelos de base de datos adicionales se almacenan en este paquete, como un módulo separado, y se agregan en __init__.py.
Agreguemos un modelo de estudiante que configurará una tabla de estudiantes en nuestro sqlite base de datos.
Hello\hello\model\student.py
from sqlalchemy import *
from sqlalchemy.orm import mapper, relation, relation, backref
from sqlalchemy import Table, ForeignKey, Column
from sqlalchemy.types import Integer, Unicode, DateTime
from hello.model import DeclarativeBase, metadata, DBSession
from datetime import datetime
class student(DeclarativeBase):
__tablename__ = 'student'
uid = Column(Integer, primary_key = True)
name = Column(Unicode(20), nullable = False, default = '')
city = Column(Unicode(20), nullable = False, default = '')
address = Column(Unicode(100), nullable = False, default = '')
pincode = Column(Unicode(10), nullable = False, default = '')
Ahora agregue este modelo en init_model() funcionar dentro __init__.py.Esta función ya contiene el modelo de autenticación. Agregue nuestro modelo de estudiante debajo.
# Import your model modules here.
from hello.model.auth import User, Group, Permission
from hello.model.student import student
Si desea que la tabla se inicialice con algunos datos al momento de configurar los modelos, agréguela en bootstrap.py en el paquete websetup. Agregue las siguientes declaraciones en elbootstrap() función.
s1 = model.student()
s1.name = 'M.V.Lathkar'
s1.city = 'Nanded'
s1.address = 'Shivaji Nagar'
s1.pincode = '431602'
model.DBSession.add(s1)
model.DBSession.flush()
transaction.commit()
Los modelos se inicializan ejecutando el comando setup-app de la caja de cambios -
gearbox setup-app
El objeto de sesión de SQLAlchemy gestiona todas las operaciones de persistencia del objeto ORM.
Los siguientes métodos de sesión realizan operaciones CRUD:
DBSession.add(model object) : Inserta un registro en la tabla asignada.
DBSession.delete(model object) - elimina el registro de la tabla.
DBSession.query(model).all() - recupera todos los registros de la tabla (correspondientes a una consulta SELECT).
Puede aplicar un filtro al conjunto de registros recuperados mediante un atributo de filtro. Por ejemplo, para recuperar registros con city = 'Hyderabad' en la tabla de estudiantes, use la siguiente declaración:
DBSession.query(model.student).filter_by(city = ’Hyderabad’).all()
Ahora veremos cómo interactuar con los modelos a través de las URL del controlador.
Primero, diseñemos un formulario ToscaWidgets para ingresar los datos del estudiante
Hello\hello\controllers.studentform.py
import tw2.core as twc
import tw2.forms as twf
class StudentForm(twf.Form):
class child(twf.TableLayout):
name = twf.TextField(size = 20)
city = twf.TextField()
address = twf.TextArea("",rows = 5, cols = 30)
pincode = twf.NumberField()
action = '/save_record'
submit = twf.SubmitButton(value = 'Submit')
En RootController (root.py de la aplicación Hello), agregue la siguiente función de mapeo '/ agregar' URL -
from hello.controllers.studentform import StudentForm
class RootController(BaseController):
@expose('hello.templates.studentform')
def add(self, *args, **kw):
return dict(page='studentform', form = StudentForm)
Guarde el siguiente código HTML como studentform.html en la carpeta de plantillas -
<!DOCTYPE html>
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<head>
<title>Student Registration Form</title>
</head>
<body>
<div id = "getting_started">
${form.display(value = dict(title = 'Enter data'))}
</div>
</body>
</html>
Entrar http://localhost:8080/adden el navegador después de iniciar el servidor. El siguiente formulario de información del estudiante se abrirá en el navegador:
El formulario anterior está diseñado para ser enviado al ‘/save_record’URL. De ahí unsave_record() la función necesita ser agregada en el root.pypara exponerlo. Esta función recibe los datos del formulario del alumno comodict()objeto. Se utiliza para agregar un nuevo registro en la tabla de estudiantes subyacente al modelo de estudiantes.
@expose()
#@validate(form = AdmissionForm, error_handler = index1)
def save_record(self, **kw):
newstudent = student(name = kw['name'],city = kw['city'],
address = kw['address'], pincode = kw['pincode'])
DBSession.add(newstudent)
flash(message = "new entry added successfully")
redirect("/listrec")
Tenga en cuenta que después de la adición exitosa, el navegador será redirigido a ‘/listrec’ URL. Esta URL está expuesta por unlistrec() function. Esta función selecciona todos los registros en la tabla de estudiantes y los envía en forma de un objeto dict a la plantilla studentlist.html. Estalistrec() la función es la siguiente:
@expose ("hello.templates.studentlist")
def listrec(self):
entries = DBSession.query(student).all()
return dict(entries = entries)
La plantilla studentlist.html itera a través del objeto del diccionario de entradas usando la directiva py: for. La plantilla studentlist.html es la siguiente:
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/">
<head>
<link rel = "stylesheet" type = "text/css" media = "screen"
href = "${tg.url('/css/style.css')}" />
<title>Welcome to TurboGears</title>
</head>
<body>
<h1>Welcome to TurboGears</h1>
<py:with vars = "flash = tg.flash_obj.render('flash', use_js = False)">
<div py:if = "flash" py:replace = "Markup(flash)" />
</py:with>
<h2>Current Entries</h2>
<table border = '1'>
<thead>
<tr>
<th>Name</th>
<th>City</th>
<th>Address</th>
<th>Pincode</th>
</tr>
</thead>
<tbody>
<py:for each = "entry in entries">
<tr>
<td>${entry.name}</td> <td>${entry.city}</td>
<td>${entry.address}</td> <td>${entry.pincode}</td>
</tr>
</py:for>
</tbody>
</table>
</body>
</html>
Ahora revise el http://localhost:8080/adde ingrese los datos en el formulario. Al hacer clic en el botón enviar, el navegador se dirigirá a studentlist.html. También mostrará un mensaje de "nuevo registro agregado correctamente".
ToscaWidgets contiene un control DataGrid que proporciona una forma rápida de presentar datos en forma tabular. El objeto DataGrid se declara de la siguiente manera:
from tw2.forms import DataGrid
student_grid = DataGrid(fields = [('Name', 'name'),('City', 'city'),
('Address','address'), ('PINCODE', 'pincode')])
Ahora, la función showgrid () recupera todos los registros en la tabla de estudiantes y expone los datos a la plantilla grid.html. Primero, el código para la función showgrid () y luego el código grid.html se proporciona a continuación:
Mostrar cuadrícula()
@expose('hello.templates.grid')
def showgrid(self):
data = DBSession.query(student).all()
return dict(page = 'grid', grid = student_grid, data = data)
grid.html
<!DOCTYPE html>
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
lang = "en">
<head>
<title>Student Registration Form</title>
</head>
<body>
<div id = "getting_started">
<div>${grid.display(value = data)}</div>
</div>
</body>
</html>
Los siguientes datos tabulares se mostrarán cuando http://localhost:8080/showlist La URL se ingresa en el navegador -
TurboGears proporciona un decorador conveniente llamado paginate () para dividir la salida en las páginas. Este decorador se combina con el decorador expose (). El decorador @Paginate () toma el objeto de diccionario del resultado de la consulta como argumento. Además, el número de registros por página se decide por el valor del atributo items_per_page. Asegúrese de importar la función de paginación de tg.decorators en su código.
Reescriba la función listrec () en root.py de la siguiente manera:
from tg.decorators import paginate
class RootController(BaseController):
@expose ("hello.templates.studentlist")
@paginate("entries", items_per_page = 3)
def listrec(self):
entries = DBSession.query(student).all()
return dict(entries = entries)
Los elementos por página se establecen en tres.
En la plantilla studentlist.html, la navegación de la página se habilita agregando tmpl_context.paginators.entries.pager () debajo de la directiva py: for. El código para esta plantilla debe ser el siguiente:
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/">
<head>
<link rel = "stylesheet" type = "text/css" media = "screen"
href = "${tg.url('/css/style.css')}" />
<title>Welcome to TurboGears</title>
</head>
<body>
<h1>Welcome to TurboGears</h1>
<py:with vars = "flash = tg.flash_obj.render('flash', use_js = False)">
<div py:if = "flash" py:replace = "Markup(flash)" />
</py:with>
<h2>Current Entries</h2>
<table border = '1'>
<thead>
<tr>
<th>Name</th>
<th>City</th>
<th>Address</th>
<th>Pincode</th>
</tr>
</thead>
<tbody>
<py:for each = "entry in entries">
<tr>
<td>${entry.name}</td> <td>${entry.city}</td>
<td>${entry.address}</td> <td>${entry.pincode}</td>
</tr>
</py:for>
<div>${tmpl_context.paginators.entries.pager()}</div>
</tbody>
</table>
</body>
</html>
Entrar http://localhost:8080/listrecen el navegador. Se muestra la primera página de registros de la tabla. En la parte superior de esta tabla, también se ven enlaces a números de página.
Cómo agregar soporte de paginación a Datagrid
También es posible agregar soporte de paginación a datagrid. En el siguiente ejemplo, la cuadrícula de datos paginada está diseñada para mostrar el botón de acción. Para activar el botón de acción, el objeto de cuadrícula de datos se construye con el siguiente código:
student_grid = DataGrid(fields = [('Name', 'name'),('City', 'city'),
('Address','address'), ('PINCODE', 'pincode'),
('Action', lambda obj:genshi.Markup('<a
href = "%s">Edit</a>' % url('/edit',
params = dict(name = obj.name)))) ])
Aquí, el botón de acción está vinculado al parámetro de nombre de cada fila en la cuadrícula de datos.
Reescribir el showgrid() funciona de la siguiente manera:
@expose('hello.templates.grid')
@paginate("data", items_per_page = 3)
def showgrid(self):
data = DBSession.query(student).all()
return dict(page = 'grid', grid = student_grid, data = data)
El navegador muestra la cuadrícula de datos paginada de la siguiente manera:
Al hacer clic en el botón Editar en la tercera fila, se redirigirá a la siguiente URL http://localhost:8080/edit?name=Rajesh+Patil
TurboGears proporciona la extensión tgext.admin, que funciona con tgext.crud y sprox. Este Sprox es un paquete que se utiliza para la creación de widgets web directamente desde el esquema de la base de datos. Esto se puede usar para crear automáticamente páginas de administración simples y es el kit de herramientas que alimenta la página / admin en las aplicaciones recién iniciadas.
De forma predeterminada, el administrador proporcionará un acceso generado automáticamente a todos los modelos importados en los modelos de su proyecto / __ init__.py.
Cómo crear un administrador de TurboGears
El administrador de TurboGears predeterminado se crea como un objeto de la clase AdminController:
from tgext.admin.controller import AdminController
class RootController(BaseController):
admin = AdminController(model, DBSession, config_type = TGAdminConfig)
Esto crea un administrador para todos los modelos con la configuración de administrador de TurboGears predeterminada.
A través del administrador, se ha creado un usuario durante la fase de configuración. Ahora, es posible obtener acceso al administrador de TurboGears enhttp://localhost:8080/adminLa primera vez que acceda a esta página, le pedirá autenticación. Simplemente puede proporcionar el nombre de usuario y la contraseña del usuario que el comando setup-app creó para nosotros:
Username: manager
Password: managepass
Para iniciar sesión en el proyecto de inicio rápido, agregue las siguientes funciones a la clase RootController (controllers / root.py).
from hello.lib.base import BaseController
from tg import expose, flash, redirect, request,url, lurl
from tg import redirect, validate
from hello import model
from hello.model import DBSession
from tgext.admin.tgadminconfig import BootstrapTGAdminConfig as TGAdminConfig
from tgext.admin.controller import AdminController
from tg.exceptions import HTTPFound
class RootController(BaseController):
admin = AdminController(model, DBSession, config_type = TGAdminConfig)
@expose('hello.templates.index')
def index(self):
return dict(page = 'index')
@expose('hello.templates.login')
def login(self, came_from = lurl('/'), failure = None, login = ''):
if failure is not None:
if failure == 'user-not-found':
flash(_('User not found'), 'error')
elif failure == 'invalid-password':
flash(_('Invalid Password'), 'error')
login_counter = request.environ.get('repoze.who.logins', 0)
if failure is None and login_counter > 0:
flash(_('Wrong credentials'), 'warning')
return dict(page = 'login', login_counter = str(login_counter),
came_from = came_from, login = login)
@expose()
def post_login(self, came_from = lurl('/')):
if not request.identity:
login_counter = request.environ.get('repoze.who.logins', 0) + 1
redirect('/login', params = dict(came_from = came_from,
__logins = login_counter))
userid = request.identity['repoze.who.userid']
flash(('Welcome back, %s!') % userid)
return HTTPFound(location = came_from)
Inicie sesión en la aplicación de 'inicio rápido' después de iniciar el servidor y visitando http://localhost:8080/loginy luego ingrese las credenciales de administrador como se muestra arriba. El navegador mostrará una página de administración como la que se muestra a continuación:
La página muestra todos los modelos creados en esta aplicación. Puede hacer clic en cualquier modelo para ver la lista de entradas en él:
El botón 'Nuevo' en la parte superior de esta cuadrícula de datos permite agregar el registro. De manera similar, los botones de acción para editar y eliminar un registro también se proporcionan en la columna de acciones de esta cuadrícula de datos. También se muestra un cuadro de búsqueda para seleccionar registros de forma condicional.
Una aplicación TurboGears se crea mediante las opciones de inicio rápido y configuración de la aplicación del kit de herramientas de la caja de herramientas, que tiene la autorización y el soporte de autenticación habilitado de forma predeterminada. Los modelos declarados en auth.py se configuran e inicializan según los valores asignados en bootstrap.py.
Los siguientes modelos están declarados en auth.py:
Modelo de usuario
El modelo de usuario contiene el diseño de una tabla tg_user. Esta tabla es utilizada por el paquete repose.who. Este paquete repose.who es una biblioteca de autenticación potente y extensible para aplicaciones WSGI. La estructura de un modelo de usuario es la siguiente:
class User(DeclarativeBase):
"""
__tablename__ = 'tg_user'
user_id = Column(Integer, autoincrement = True, primary_key=True)
user_name = Column(Unicode(16), unique = True, nullable = False)
email_address = Column(Unicode(255), unique = True,nullable=False)
display_name = Column(Unicode(255))
_password = Column('password', Unicode(128))
created = Column(DateTime, default = datetime.now)
Este modelo de grupo contiene la tabla de definición tg_group. Su definición se da en auth.py de la siguiente manera:
class Group(DeclarativeBase):
__tablename__ = 'tg_group'
group_id = Column(Integer, autoincrement = True,primary_key = True)
group_name = Column(Unicode(16),unique = True,nullable = False)
display_name = Column(Unicode(255))
created = Column(DateTime, default = datetime.now)
También se configura otro permiso modelo, que contiene la definición del permiso.
class Permission(DeclarativeBase):
__tablename__ = 'tg_permission'
permission_id = Column(Integer,autoincrement = True,primary_key = True)
permission_name = Column(Unicode(63), unique = True, nullable = False)
description = Column(Unicode(255))
En el momento de configurar los modelos, se agregan los siguientes datos en estas tablas:
u = model.User()
u.user_name = 'manager'
u.display_name = 'Example manager'
u.email_address = '[email protected]'
u.password = 'managepass'
model.DBSession.add(u)
g = model.Group()
g.group_name = 'managers'
g.display_name = 'Managers Group'
g.users.append(u)
model.DBSession.add(g)
p = model.Permission()
p.permission_name = 'manage'
p.description = 'This permission gives an administrative right'
p.groups.append(g)
model.DBSession.add(p)
u1 = model.User()
u1.user_name = 'editor'
u1.display_name = 'Example editor'
u1.email_address = '[email protected]'
u1.password = 'editpass'
model.DBSession.add(u1)
Modelo de predicado
El módulo de predicados en el paquete tg contiene definiciones para verificadores de predicados. Un predicado es una condición que debe cumplirse para que el usuario pueda acceder a la fuente solicitada. Tal predicado, o condición, puede estar formado por más predicados, a los que se les llama predicados compuestos. Los controladores de acción, o controladores, pueden tener solo un predicado, ya sea simple o compuesto.
Si un usuario no ha iniciado sesión, o no tiene los permisos adecuados, este verificador de predicados arroja un 401 (HTTP no autorizado), que es capturado por el middleware repoze.who para mostrar la página de inicio de sesión que permite al usuario iniciar sesión y redirigir el el usuario vuelve a la página adecuada cuando haya terminado.
Las diferentes condiciones o predicados definidos en el módulo tg.predicates son:
No Señor. | tg.predicates módulo y descripción |
---|---|
1 | All Compruebe si se cumplen todos los predicados especificados |
2 | Any Compruebe si se cumple al menos uno de los predicados especificados |
3 | is_user Verifique que el nombre de usuario del usuario autenticado sea el especificado |
4 | in_group Compruebe que el usuario pertenece al grupo específico. |
5 | in_all_groups Compruebe que el usuario pertenece a todos los grupos especificados. |
6 | in_any_group Compruebe que el usuario pertenece al menos a uno de los grupos especificados. |
7 | is_anonymous Compruebe que el usuario actual sea anónimo. |
8 | has_permission Compruebe que el usuario actual tenga el permiso especificado. |
9 | has_all_permissions Compruebe que al usuario actual se le hayan otorgado todos los permisos especificados. |
10 | has_any_permission Compruebe que el usuario tenga al menos uno de los permisos especificados. |
Por ejemplo, si tiene un predicado, que es grant access user belonging to customers group, entonces puede usar el siguiente verificador de predicados incorporado:
from tg.predicates import in_group
p in_group(‘customers’)
El siguiente verificador de predicados otorgará acceso al usuario 'root' o cualquier persona con permiso de 'administrar' -
from tg.predicates import Any, is_user, has_permission
p = Any(is_user('root'), has_permission('manage'),
sg = 'Only administrators can remove blog posts')
TurboGears también admite bases de datos de documentos MongoDB. Utiliza Ming, una API Object Document Mapper. El uso de Ming es muy similar al de SQLAlchemy. El lenguaje de consulta Ming hace posible trasladar el proyecto TurboGears basado en SQLAlchemy a Ming.
Que es PyMongo
PyMongo es una distribución de Python que contiene herramientas para trabajar con MongoDB. Ming extiende PyMongo proporcionando:
- Modelos declarativos
- Validación y conversión de esquemas
- Evolución del esquema
- Implementación de Pure InMemory MongoDB
- Unidad de trabajo
- Mapa de identidad
- Relaciones uno a varios, varios a uno y varios a varios
En primer lugar, debe descargar e instalar MongoDB. La última distribución de MongoDB se puede descargar desdehttps://www.mongodb.org/downloads
En Windows, inicie el servidor MongoDB proporcionando la opción -dbpath -
C:\mongodb\bin>Mongod --dbpath d:\mongo
D:\mongo folderestá designado para almacenar la base de datos MongoDB. El servidor comienza a escuchar enhttp://localhost:27017. Ahora, para iniciar el shell de MongoDB, use el siguiente comando:
C:\mongodb\bin>Mongo
Nuestro entorno MongoDB ya está listo.
Ahora cree un proyecto TurboGears con la opción -ming -
gearbox quickstart --ming Hello
Este proyecto de inicio rápido proporcionará una capa de autenticación y autorización como la que se proporciona para la versión SQLAlchemy. Esta aplicación ahora intentará conectarse a un servidor en el puerto 27017 de la máquina local. El archivo development.ini en la carpeta del proyecto contiene la siguiente configuración:
ming.url = mongodb://localhost:27017/
ming.db = hello
Configure el proyecto usando el siguiente comando:
Python setup.py develop
La carpeta del proyecto contiene la subcarpeta de modelos que tiene los siguientes archivos:
__init__.py - Aquí es donde el databaseel acceso está configurado. Tus colecciones deben serimported into this module. Por ejemplo, agregaremos la colección de estudiantes en este paquete.
session.py - Este archivo define el session of your database connection. Deberá importar esto cada vez que tenga que declarar unMappedClass para especificar la sesión a perform queries.
auth.py - Este archivo se creará, si tiene enabled authentication and authorizationen el inicio rápido. Define tres coleccionesrepoze.who, que además se basa en: Usuario, Grupo y Permiso.
Definición de su colección
De forma predeterminada, TurboGears configura Ming en modo declarativo. Esto es similar al soporte declarativo de SQLAlchemy y necesita que cada modelo herede de la clase MappedClass.
MappedClass requiere que una subclase __mongometa__ esté disponible en el interior, que además proporciona los detalles sobre el nombre de la colección que almacena los documentos y la sesión utilizada para almacenar los documentos.
MappedClass también contiene la definición de campos en el documento. El módulo odm de Ming tiene definiciones de diferentes tipos de propiedades de campo:
- FieldProperty
- ForeignIdProperty
- RelationProperty
los ming.schema module define los siguientes tipos de datos:
- ming.schema.Anything
- ming.schema.Array
- ming.schema.Binary
- ming.schema.Bool
- ming.schema.Float
- ming.schema.Int
- ming.schema.ObjectId
- ming.schema.Scalar
- ming.schema.String
Para agregar la colección de estudiantes en este modelo, guarde el siguiente código como student.py en la carpeta hello / models.
Hola \ modelos \ estudiante.py
from ming import schema
from ming.odm import MappedClass
from ming.odm import FieldProperty, ForeignIdProperty
from hello.model import DBSession
Class student(MappedClass):
class __mongometa__:
session = DBSession
name = 'student'
_id = FieldProperty(schema.ObjectId)
name = FieldProperty(schema.String(required = True))
city = FieldProperty(schema.String(if_missing = ''))
address = FieldProperty(schema.String(if_missing = ''))
pincode = FieldProperty(schema.String(if_missing = ''))
Finalmente, incluya este modelo en hello \ models \ __ init__.py
# Import your model modules here.
from hello.model.auth import User, Group, Permission
from hello.model.student import student
Para configurar estos modelos, ejecute el siguiente comando de caja de cambios:
Gearbox setup-app
Inicie el servidor con el siguiente comando de caja de cambios:
Gearbox serve –reload –debug
Abra la página de inicio de esta aplicación. (http://localhost:8080/)e inicie sesión con credenciales de administrador. La página de administración de esta aplicación mostrará la lista de modelos configurados. (iniciar sesión como administrador, administrar contraseña)
La creación de colecciones también se puede verificar en la interfaz web de MongoDB, así como en el shell de MongoDB.
La ODMSession se utiliza para realizar varias operaciones de base de datos utilizando las siguientes funciones:
- model.query.find()
- model.query.find_and_modify()
- model.remove()
- model.update()
- model.flush()
Diseñar un formulario ToscoWidget
Ahora diseñaremos un formulario ToscoWidget para ingresar datos de los estudiantes y agregarlos a la tabla subyacente al modelo del estudiante.
A continuación se muestra el código para crear un studentform.py:
Hola \ controllers \ studentform.py
import tw2.core as twc
import tw2.forms as twf
class StudentForm(twf.Form):
class child(twf.TableLayout):
name = twf.TextField(size = 20)
city = twf.TextField()
address = twf.TextArea("",rows = 5, cols = 30)
pincode = twf.NumberField()
action = '/save_record'
submit = twf.SubmitButton(value = 'Submit')
En la URL de la aplicación Rootcontroller '/ add' que llama a la función add (), que abrirá el formulario diseñado anteriormente en el navegador. Su botón de envío luego invoca la función save_record (). Recupera los datos del formulario y los guarda en la tabla de estudiantes y redirige la aplicación a la URL '/ listrec', que expone la plantilla de lista de estudiantes.
El root.py para esta actividad es el siguiente:
Hola / controllers / root.py
from hello.lib.base import BaseController
from tg import expose, flash, redirect, request,url, lurl
from tg import redirect, validate
from hello import model
from hello.model import DBSession
from hello.model.student import student
from hello.controllers.studentform import StudentForm
class RootController(BaseController):
@expose()
def index(self):
return "<h1>Hello World</h1>"
@expose ("hello.templates.studentlist")
def listrec(self):
entries = student.query.find()
return dict(entries = entries)
@expose('hello.templates.studentform')
def add(self, *args, **kw):
return dict(page = 'studentform', form = StudentForm)
@expose()
def save_record(self, **kw):
newstudent = student(name = kw['name'],city = kw['city'],
address = kw['address'], pincode = kw['pincode'])
DBSession.flush()
flash(message = "new entry added successfully")
redirect("/listrec")
Las siguientes plantillas se crean en la carpeta de plantillas:
Hola \ templates \ studentform.html
<!DOCTYPE html>
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/" lang = "en">
<head>
<title>Student Registration Form</title>
</head>
<body>
<div id = "getting_started">
${form.display(value = dict(title = 'Enter data'))}
</div>
</body>
</html>
Hola \ templates \ studentlist.html
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/">
<head>
<link rel = "stylesheet" type = "text/css" media = "screen"
href = ${tg.url('/css/style.css')}" /> <title>Welcome to TurboGears</title> </head> <body> <h1>Welcome to TurboGears</h1> <py:with vars = "flash = tg.flash_obj.render('flash', use_js = False)"> <div py:if = "flash" py:replace = "Markup(flash)" /> </py:with> <h2>Current Entries</h2> <table border = '1'> <thead> <tr> <th>Name</th> <th>City</th> <th>Address</th> <th>Pincode</th> </tr> </thead> <tbody> <py:for each = "entry in entries"> <tr> <td>${entry.name}</td>
<td>${entry.city}</td> <td>${entry.address}</td>
<td>${entry.pincode}</td>
</tr>
</py:for>
</tbody>
</table>
</body>
</html>
Reinicie el servidor e ingrese http://localhost:8080/add en el navegador -
Cada vez que se agregan los datos y se presiona el botón Enviar, se mostrará la lista de entradas actuales.
El kit de herramientas de Gearbox contiene un comando de andamio, que es muy útil para crear rápidamente nuevos componentes de la aplicación TurboGears. Una aplicación generada por el comando de inicio rápido de gearbox tiene una plantilla de esqueleto en la carpeta del modelo (model.py.template), una carpeta de plantillas (template.html.template) y una carpeta de controladores (controller.py.template). Estos archivos '.template' se utilizan como base para crear nuevos andamios para una aplicación
Por ejemplo, para crear un nuevo modelo llamado mymodel, simplemente ejecute el siguiente comando:
gearbox scaffold model mymodel
Este comando generará model / mymodel.py con la clase newmodel definida en él.
# -*- coding: utf-8 -*-
"""Mymodel model module."""
from sqlalchemy import *
from sqlalchemy import Table, ForeignKey, Column
from sqlalchemy.types import Integer, Unicode, DateTime, LargeBinary
from sqlalchemy.orm import relationship, backref
from hello.model import DeclarativeBase, metadata, DBSession
class Mymodel(DeclarativeBase):
__tablename__ = 'mymodels'
uid = Column(Integer, primary_key = True)
data = Column(Unicode(255), nullable = False)
user_id = Column(Integer, ForeignKey('tg_user.user_id'), index = True)
user = relationship('User', uselist = False,
backref = backref('mymodels',cascade = 'all, delete-orphan'))
__all__ = ['Mymodel']
Los usuarios ahora pueden realizar modificaciones en la estructura de la tabla según sus requisitos y luego importarla dentro model/__init__.py para que el modelo esté disponible dentro de la aplicación.
Para crear un modelo, una clase de controlador para manejarlo y una página de índice, estos tres componentes se pueden crear simultáneamente con el siguiente comando.
gearbox scaffold model controller template mymodel
Este comando dará como resultado controllers \ mymodel.py en el que la clase MymodelController está debidamente definida.
# -*- coding: utf-8 -*-
"""Mymodel controller module"""
from tg import expose, redirect, validate, flash, url
# from tg.i18n import ugettext as _
# from tg import predicates
from hello.lib.base import BaseController
# from hello.model import DBSession
class MymodelController(BaseController):
# Uncomment this line if your controller requires an authenticated user
# allow_only = predicates.not_anonymous()
@expose('hello.templates.mymodel')
def index(self, **kw):
return dict(page = 'mymodel-index')
Para comenzar a usar este controlador, móntelo dentro de su aplicación RootController solo para definir una instancia de MymodelController. Agrega estas líneas en controllers \ root.py -
From hello.controller.mymodel import MymodelController
class RootController(BaseController): mymodel = MymodelController()
También se creará un andamio de plantilla templates \ mymodel.html en la carpeta de plantillas. Actuará como una página de índice para la URL '/ mymodel'.
El generado mymodel.html file en la carpeta de plantillas será la siguiente:
<html xmlns = "http://www.w3.org/1999/xhtml"
xmlns:py = "http://genshi.edgewall.org/"
xmlns:xi = "http://www.w3.org/2001/XInclude">
<xi:include href = "master.html" />
<head>
<title>Mymodel</title>
</head>
<body>
<div class = "row">
<div class = "col-md-12">
<h2>Mymodel</h2>
<p>Template page for Mymodel</p>
</div>
</div>
</body>
</html>
Hay tres formas en TurboGears de conectar comportamientos dentro de las aplicaciones existentes.
Hook - Es un mecanismo mediante el cual es posible definir un evento y notificar a los oyentes registrados cuando se emiten los eventos.
Controller Wrapper- Se encuentra entre TurboGears y Controller, por lo que es posible extender el controlador como un decorador. Por lo tanto, se puede adjuntar a cualquier aplicación de controlador de terceros.
Application Wrapper - Es similar a cualquier middleware WSGI, pero solo funciona en el contexto TurboGears.
Aquí, en este capítulo, discutiremos cómo usar los ganchos dentro de una aplicación existente.
Manos
Los hooks son eventos registrados en el archivo de configuración de la aplicación. app_cfg.py. A continuación, los decoradores de eventos conectan cualquier controlador a estos eventos.
Los siguientes ganchos están definidos en TurboGears:
No Señor. | Ganchos y descripción |
---|---|
1 | Startup() solo para toda la aplicación, se llama cuando se inicia la aplicación. |
2 | shutdown() solo para toda la aplicación, se llama cuando se cierra la aplicación. |
3 | configure_new_app nueva aplicación creada por el configurador de aplicaciones. |
4 | before_config(app) solo para toda la aplicación, llamado justo después de crear la aplicación, pero antes de configurar las opciones y el middleware |
5 | after_config(app) solo para toda la aplicación, se llama después de terminar de configurar todo. |
6 | before_validate Llamado antes de realizar la validación |
7 | before_call Se llama después de la validación, antes de llamar al método del controlador real. |
8 | before_render Llamado antes de renderizar una plantilla de controlador, la salida es el valor de retorno del controlador. |
9 | after_render Se llama después de terminar de renderizar una plantilla de controlador. |
Registrar un gancho
A fin de que register a Hook, crear funciones en app_cfg.py y luego regístrelos usando el siguiente código -
tg.hooks.register(hookane, function, controller)
En el siguiente código, los ganchos on_startup, on_shutdown y before_render están registrados en app_cfg.py.
def on_startup():
print 'hello, startup world'
def on_shutdown():
print 'hello, shutdown world'
def before_render(remainder, params, output):
print 'system wide before render'
# ... (base_config init code)
tg.hooks.register('startup', on_startup)
tg.hooks.register('shutdown', on_shutdown)
tg.hooks.register('before_render', before_render)
El gancho before_render se registra con una función de controlador en Rootcontroller. Agregue el siguiente código en controllers \ root.py.
from tg.decorators import before_render
class RootController(BaseController):
@expose('hello.templates.index')
@before_render(before_render_cb)
def index(self, *args, **kw):
return dict(page = 'index')
Cuando se sirve la aplicación, se muestra el mensaje de inicio en la consola.
hello, startup world
Starting Standard HTTP server on http://127.0.0.1:8080
Cuando se ingresa la URL '/' en el navegador, se muestra en la consola un mensaje correspondiente al gancho before_render.
system wide before render
Going to render {'page': 'index'}
Las extensiones de TurboGears se identifican mediante tgext.* package. Un kit de herramientas de Gearbox proporciona un comando tgext para crear una extensión de muestra. Por ejemplo
gearbox tgext -n myextension
Otros parámetros opcionales para este comando son:
--author - nombre del autor del paquete.
--email - correo electrónico del autor del paquete.
--licence- licencia utilizada para el paquete. El valor predeterminado es MIT.
--description - Descripción del paquete.
--keywords - Palabras clave del paquete (predeterminado: turbogears2.extension).
Esto creará un directorio tgext.myextension, que tiene una extensión de muestra simple dentro.
Run the setup.py dentro del directorio -
Python setup.py install
los _init_.py archivo dentro tgext/myextension carpeta contiene -
Plugme function - Este es el punto de entrada de la extensión.
SetupExtension class - La inicialización de la extensión tiene lugar aquí.
On_startup function - dentro de la clase hay un gancho registrado en la función __call__ dentro de la clase.
Breve versión del tgext\myextension\__init__.py.
from tg import config
from tg import hooks
from tg.configuration import milestones
import logging
log = logging.getLogger('tgext.myextension')
def plugme(configurator, options = None):
if options is None:
options = {}
log.info('Setting up tgext.myextension extension...')
milestones.config_ready.register(SetupExtension(configurator))
return dict(appid='tgext.myextension')
class SetupExtension(object):
def __init__(self, configurator):
self.configurator = configurator
def __call__(self):
log.info('>>> Public files path is %s' % config['paths']['static_files'])
hooks.register('startup', self.on_startup)
def echo_wrapper_factory(handler, config):
def echo_wrapper(controller, environ, context):
log.info('Serving: %s' % context.request.path)
return handler(controller, environ, context)
return echo_wrapper
self.configurator.register_wrapper(echo_wrapper_factory)
def on_startup(self):
log.info('+ Application Running!')
Una vez que la extensión está instalada, actívela haciendo las siguientes adiciones en la aplicación app_cfg.py archivo de configuración.
from tgext.myextension import plugme
plugme(base_config)
Si iniciamos el servidor usando un comando del servidor de caja de cambios, la notificación de una extensión recién registrada se puede ver en la consola de la siguiente manera:
14:29:13,250 INFO [tgext.myextension] Setting up tgext.myextension extension...
14:29:13,453 INFO [tgext.myextension] >>> Public files path is c:\tghello\hello\hello\public
14:29:13,453 INFO [tgext.myextension] + Application Running!
Starting Standard HTTP server on http://127.0.0.1:8080
Si su extensión necesita exponer modelos y controladores, probablemente desee echar un vistazo a la Pluggable Applications, que están destinadas a crear aplicaciones Turbogears reutilizables que se pueden conectar dentro de otras aplicaciones para ampliar sus funciones.
Utilice el siguiente comando de caja de cambios para crear una aplicación conectable:
gearbox quickstart-pluggable plugtest
Estas aplicaciones conectables pueden definir las suyas propias:
Controllers - que se montará automáticamente cuando se purgue la aplicación.
Models - que estará disponible dentro y fuera de la aplicación conectada.
Helpers - que se puede exponer automáticamente en el objeto 'H' en una plantilla de aplicación.
Bootstrap - que se ejecutará cuando se llame a setup-app.
Statics - que estarán disponibles en su propia ruta privada.
Instale esta aplicación plugtest y monte la misma haciendo las siguientes modificaciones en app_cfg.py.
from tgext.pluggable import plug
plug(base_config, plugtest)
REST significa REpresentacional State Ttransferir. REST es una arquitectura basada en estándares web y utiliza el protocolo HTTP para la comunicación de datos. Gira en torno a un recurso donde cada componente es un recurso y se accede a un recurso mediante una interfaz común utilizando métodos estándar HTTP. REST fue introducido por primera vez porRoy Fielding in 2000.
¿Qué es un RestController?
RestController en TurboGears proporciona un mecanismo para acceder al método de la solicitud, no solo a la URL. La verborrea HTTP estándar incluye: GET, POST, PUT y DELETE. RestController los admite, y también agrega algunos accesos directos para el envío de URL que hacen que mostrar los datos como formularios y listas, sea un poco más fácil para el usuario.
Para explicar cómo funciona RESTful con TurboGears, vamos a definir un servicio web simple que expone una lista de estudiantes.
El código para el modelo de estudiante se proporciona a continuación:
modelo \ student.py
# -* - coding: utf-8 -*-
from sqlalchemy import *
from sqlalchemy.orm import mapper, relation, relation, backref
from sqlalchemy import Table, ForeignKey, Column
from sqlalchemy.types import Integer, Unicode, DateTime
from hello.model import DeclarativeBase, metadata, DBSession
from datetime import datetime
class student(DeclarativeBase):
__tablename__ = 'student'
uid = Column(Integer, primary_key = True)
name = Column(Unicode(20), nullable = False, default = '')
city = Column(Unicode(20), nullable = False, default = '')
address = Column(Unicode(100), nullable = False, default = '')
pincode = Column(Unicode(10), nullable = False, default = '')
Ahora cree un controlador basado en RestController y proporcione una función de vista para enumerar la lista de estudiantes en formato json.
Controladores \ student.py
from tg import RestController
from tg import expose
from hello import model
from hello.model import DBSession
from hello.model.student import student
from tg.decorators import with_trailing_slash
class StudentController(RestController):
@expose('json')
def get_all(self):
students = DBSession.query(student).all()
return dict(students=students)
Monte este StudentController en RootController de la aplicación incorporando las siguientes líneas en root.py -
from hello.controllers.student import StudentController
class RootController(BaseController):
students = StudentController()
Yendo al http://localhost:8080/students proporcionará la lista de nuestros estudiantes codificada en formato json.
Usamos el método de publicación para definir cómo guardamos a nuestro estudiante en la base de datos. Este método se llama siempre que el http://localhost:8080/student se accede a la URL mediante una solicitud POST -
@expose('json')
def post(self, name, city, address, pincode):
newstudent = student(name = name, city = city, address = address, pincode = pincode)
DBSession.add(newstudent)
DBSession.flush()
return dict(student = newstudent)
Utilizando el get_one() método, podemos mostrar un elemento de la base de datos al usuario -
@expose('json')
def get_one(self, movie_id):
newstudent = DBSession.query(student).get(uid)
return dict(movie = movie)
PUT es el método utilizado para actualizar un registro existente usando REST -
@expose('json')
def put(self, name = name, city = city, address = address, pincode = pincode, **kw):
newstudent = DBSession.query(student).get(name)
newstudent.name = name
newstudent.city = city
newstudent.address = address
newstudent.pincode = pincode
return dict(student = newstudent)
El caballo de batalla de delete se adjunta al método post_delete. Aquí, de hecho, eliminamos el registro de la base de datos y luego lo redirigimos a la página de listado:
@expose('json')
def post_delete(self, uid, **kw):
newstudent = DBSession.query(student).get(uid)
DBSession.delete(newstudent)
return dict(movie = newstudent.uid)
Para pasar de un entorno de desarrollo a un entorno de producción completo, la aplicación debe implementarse en un servidor web real. Dependiendo de lo que tenga, existen diferentes opciones disponibles para implementar una aplicación web TurboGears.
Apache con mod_wsgi
El mod_wsgi es un módulo de Apache desarrollado por Graham Dumpleton. Permite que los programas WSGI se sirvan utilizando el servidor web Apache.
En primer lugar, instale Apache 2.X para su plataforma, si aún no lo ha hecho. Una vez que haya instalado Apache, instale mod_wsgi. Cree y active el entorno virtual Python en el servidor e instale TurboGears en él.
Instale su aplicación dentro del director de la aplicación, luego cree un script llamado app.wsgi.
Configure la instalación de Apache de la siguiente manera:
<VirtualHost *:80>
ServerName www.site1.com
WSGIProcessGroup www.site1.com
WSGIDaemonProcess www.site1.com user = <username>
group = www-data threads = 4 python-path = <pythonpath>
WSGIScriptAlias myapp/app.wsgi
#Serve static files directly without TurboGears
Alias /images
Alias /css
Alias /js
CustomLog
ErrorLog
</VirtualHost>
Reiniciar Apache
Tipo http://www.site1.com/ en un navegador para acceder a la aplicación.
TurboGears bajo Circus y Chaussette
Circus es un administrador de procesos y conexiones. Puede usarse para monitorear y controlar procesos y sockets. Cuando se combina con el servidor Chaussette WSGI, puede convertirse en una herramienta poderosa para implementar su aplicación y administrar cualquier proceso relacionado que sus aplicaciones necesiten.
TurboGears - GoogleAppEngine
Instale el SDK de Google AppEngine para Python desde la siguiente URL: https://cloud.google.coms
Instale Google AppEngine en su sistema. Luego, abra la consola de desarrollador de Google e inicie sesión con su cuenta de Google:https://console.developers.google.com/start
Crea un nuevo proyecto llamado mytgapp -
Con Google AppEngine Launcher, cree una nueva aplicación denominada mytgapp.
Los siguientes archivos se crearán en el directorio especificado:
- app.yaml
- favicon.ico
- index.yaml
- main.py
De forma predeterminada, la aplicación creada se basa en el marco Webapp2. Para eliminar esta dependencia, edite el archivo app.yaml y elimine la siguiente parte:
libraries:
- name: webapp2
version: "2.5.2"
Cree un entorno virtual temporal en un directorio llamado mytgapp e instale TurboGears. Cree una aplicación TurboGears en él. Ahora podemos continuar editando elmain.py que se inicia con AppEngine para ejecutar nuestra aplicación y escribir una aplicación TurboGears allí.
Agregue el siguiente contenido en main.py -
import os
import site
site.addsitedir(os.path.join(os.path.dirname(__file__), 'packages'))
from tg import expose, TGController, AppConfig
class RootController(TGController):
@expose()
def index(self):
return "<h1>Hello World</h1>"
config = AppConfig(minimal = True, root_controller = RootController())
app = config.make_wsgi_app()
Ahora ejecute la aplicación desde AppEngine Launcher y haga clic en el botón Examinar para ver que la aplicación funciona correctamente en el host local.
Ya hemos creado un proyecto llamado mytgapp en la consola del desarrollador. Ahora haga clic en el botón de implementación en el Lanzador. Una vez finalizado el proceso de implementación,http://mytgapp.appspot.com/ visite para ver nuestra solicitud en línea.