import oilarchive
This commit is contained in:
commit
2bb49ff965
37 changed files with 1027 additions and 0 deletions
4
README.txt
Normal file
4
README.txt
Normal file
|
@ -0,0 +1,4 @@
|
|||
oilarchive
|
||||
|
||||
This is a TurboGears (http://www.turbogears.org) project. It can be
|
||||
started by running the start-oilarchive.py script.
|
65
dev.cfg
Normal file
65
dev.cfg
Normal file
|
@ -0,0 +1,65 @@
|
|||
[global]
|
||||
# This is where all of your settings go for your development environment
|
||||
# Settings that are the same for both development and production
|
||||
# (such as template engine, encodings, etc.) all go in
|
||||
# oilarchive/config/app.cfg
|
||||
|
||||
# DATABASE
|
||||
|
||||
# pick the form for your database
|
||||
# sqlobject.dburi="postgres://username@hostname/databasename"
|
||||
# sqlobject.dburi="mysql://username:password@hostname:port/databasename"
|
||||
# sqlobject.dburi="sqlite:///file_name_and_path"
|
||||
|
||||
# If you have sqlite, here's a simple default to get you started
|
||||
# in development
|
||||
sqlobject.dburi="sqlite://%(current_dir_uri)s/devdata.sqlite"
|
||||
|
||||
|
||||
# if you are using a database or table type without transactions
|
||||
# (MySQL default, for example), you should turn off transactions
|
||||
# by prepending notrans_ on the uri
|
||||
# sqlobject.dburi="notrans_mysql://username:password@hostname:port/databasename"
|
||||
|
||||
# for Windows users, sqlite URIs look like:
|
||||
# sqlobject.dburi="sqlite:///drive_letter:/path/to/file"
|
||||
|
||||
# SERVER
|
||||
|
||||
# Some server parameters that you may want to tweak
|
||||
# server.socket_port=8080
|
||||
|
||||
# Enable the debug output at the end on pages.
|
||||
# log_debug_info_filter.on = False
|
||||
|
||||
server.environment="development"
|
||||
autoreload.package="oilarchive"
|
||||
|
||||
# session_filter.on = True
|
||||
|
||||
# Set to True if you'd like to abort execution if a controller gets an
|
||||
# unexpected parameter. False by default
|
||||
tg.strict_parameters = True
|
||||
|
||||
# LOGGING
|
||||
# Logging configuration generally follows the style of the standard
|
||||
# Python logging module configuration. Note that when specifying
|
||||
# log format messages, you need to use *() for formatting variables.
|
||||
# Deployment independent log configuration is in oilarchive/config/log.cfg
|
||||
[logging]
|
||||
|
||||
[[loggers]]
|
||||
[[[oilarchive]]]
|
||||
level='DEBUG'
|
||||
qualname='oilarchive'
|
||||
handlers=['debug_out']
|
||||
|
||||
[[[allinfo]]]
|
||||
level='INFO'
|
||||
handlers=['debug_out']
|
||||
|
||||
[[[access]]]
|
||||
level='INFO'
|
||||
qualname='turbogears.access'
|
||||
handlers=['access_out']
|
||||
propagate=0
|
15
oilarchive.egg-info/PKG-INFO
Normal file
15
oilarchive.egg-info/PKG-INFO
Normal file
|
@ -0,0 +1,15 @@
|
|||
Metadata-Version: 1.0
|
||||
Name: oilarchive
|
||||
Version: 1.0
|
||||
Summary: UNKNOWN
|
||||
Home-page: UNKNOWN
|
||||
Author: UNKNOWN
|
||||
Author-email: UNKNOWN
|
||||
License: UNKNOWN
|
||||
Description: UNKNOWN
|
||||
Platform: UNKNOWN
|
||||
Classifier: Development Status :: 3 - Alpha
|
||||
Classifier: Operating System :: OS Independent
|
||||
Classifier: Programming Language :: Python
|
||||
Classifier: Topic :: Software Development :: Libraries :: Python Modules
|
||||
Classifier: Framework :: TurboGears
|
21
oilarchive.egg-info/SOURCES.txt
Normal file
21
oilarchive.egg-info/SOURCES.txt
Normal file
|
@ -0,0 +1,21 @@
|
|||
README.txt
|
||||
setup.py
|
||||
start-oilarchive.py
|
||||
oilarchive/__init__.py
|
||||
oilarchive/controllers.py
|
||||
oilarchive/json.py
|
||||
oilarchive/model.py
|
||||
oilarchive/release.py
|
||||
oilarchive.egg-info/PKG-INFO
|
||||
oilarchive.egg-info/SOURCES.txt
|
||||
oilarchive.egg-info/dependency_links.txt
|
||||
oilarchive.egg-info/not-zip-safe
|
||||
oilarchive.egg-info/paster_plugins.txt
|
||||
oilarchive.egg-info/requires.txt
|
||||
oilarchive.egg-info/sqlobject.txt
|
||||
oilarchive.egg-info/top_level.txt
|
||||
oilarchive/config/__init__.py
|
||||
oilarchive/templates/__init__.py
|
||||
oilarchive/tests/__init__.py
|
||||
oilarchive/tests/test_controllers.py
|
||||
oilarchive/tests/test_model.py
|
1
oilarchive.egg-info/dependency_links.txt
Normal file
1
oilarchive.egg-info/dependency_links.txt
Normal file
|
@ -0,0 +1 @@
|
|||
|
1
oilarchive.egg-info/not-zip-safe
Normal file
1
oilarchive.egg-info/not-zip-safe
Normal file
|
@ -0,0 +1 @@
|
|||
|
2
oilarchive.egg-info/paster_plugins.txt
Normal file
2
oilarchive.egg-info/paster_plugins.txt
Normal file
|
@ -0,0 +1,2 @@
|
|||
TurboGears
|
||||
PasteScript
|
1
oilarchive.egg-info/requires.txt
Normal file
1
oilarchive.egg-info/requires.txt
Normal file
|
@ -0,0 +1 @@
|
|||
TurboGears >= 1.0
|
2
oilarchive.egg-info/sqlobject.txt
Normal file
2
oilarchive.egg-info/sqlobject.txt
Normal file
|
@ -0,0 +1,2 @@
|
|||
db_module=oilarchive.model
|
||||
history_dir=$base/oilarchive/sqlobject-history
|
1
oilarchive.egg-info/top_level.txt
Normal file
1
oilarchive.egg-info/top_level.txt
Normal file
|
@ -0,0 +1 @@
|
|||
oilarchive
|
0
oilarchive/__init__.py
Normal file
0
oilarchive/__init__.py
Normal file
0
oilarchive/config/__init__.py
Normal file
0
oilarchive/config/__init__.py
Normal file
127
oilarchive/config/app.cfg
Normal file
127
oilarchive/config/app.cfg
Normal file
|
@ -0,0 +1,127 @@
|
|||
[global]
|
||||
# The settings in this file should not vary depending on the deployment
|
||||
# environment. dev.cfg and prod.cfg are the locations for
|
||||
# the different deployment settings. Settings in this file will
|
||||
# be overridden by settings in those other files.
|
||||
|
||||
# The commented out values below are the defaults
|
||||
|
||||
# VIEW
|
||||
|
||||
# which view (template engine) to use if one is not specified in the
|
||||
# template name
|
||||
# tg.defaultview = "kid"
|
||||
|
||||
# The following kid settings determine the settings used by the kid serializer.
|
||||
|
||||
# One of (html|html-strict|xhtml|xhtml-strict|xml|json)
|
||||
# kid.outputformat="html"
|
||||
|
||||
# kid.encoding="utf-8"
|
||||
|
||||
# The sitetemplate is used for overall styling of a site that
|
||||
# includes multiple TurboGears applications
|
||||
# tg.sitetemplate="<packagename.templates.templatename>"
|
||||
|
||||
# Allow every exposed function to be called as json,
|
||||
# tg.allow_json = False
|
||||
|
||||
# List of Widgets to include on every page.
|
||||
# for exemple ['turbogears.mochikit']
|
||||
# tg.include_widgets = []
|
||||
|
||||
# Set to True if the scheduler should be started
|
||||
# tg.scheduler = False
|
||||
|
||||
# VISIT TRACKING
|
||||
# Each visit to your application will be assigned a unique visit ID tracked via
|
||||
# a cookie sent to the visitor's browser.
|
||||
# --------------
|
||||
|
||||
# Enable Visit tracking
|
||||
visit.on=True
|
||||
|
||||
# Number of minutes a visit may be idle before it expires.
|
||||
# visit.timeout=20
|
||||
|
||||
# The name of the cookie to transmit to the visitor's browser.
|
||||
# visit.cookie.name="tg-visit"
|
||||
|
||||
# Domain name to specify when setting the cookie (must begin with . according to
|
||||
# RFC 2109). The default (None) should work for most cases and will default to
|
||||
# the machine to which the request was made. NOTE: localhost is NEVER a valid
|
||||
# value and will NOT WORK.
|
||||
# visit.cookie.domain=None
|
||||
|
||||
# Specific path for the cookie
|
||||
# visit.cookie.path="/"
|
||||
|
||||
# The name of the VisitManager plugin to use for visitor tracking.
|
||||
visit.manager="sqlobject"
|
||||
|
||||
# Database class to use for visit tracking
|
||||
visit.soprovider.model = "oilarchive.model.Visit"
|
||||
|
||||
# IDENTITY
|
||||
# General configuration of the TurboGears Identity management module
|
||||
# --------
|
||||
|
||||
# Switch to turn on or off the Identity management module
|
||||
identity.on=True
|
||||
|
||||
# [REQUIRED] URL to which CherryPy will internally redirect when an access
|
||||
# control check fails. If Identity management is turned on, a value for this
|
||||
# option must be specified.
|
||||
identity.failure_url="/login"
|
||||
|
||||
# identity.provider='sqlobject'
|
||||
|
||||
# The names of the fields on the login form containing the visitor's user ID
|
||||
# and password. In addition, the submit button is specified simply so its
|
||||
# existence may be stripped out prior to passing the form data to the target
|
||||
# controller.
|
||||
# identity.form.user_name="user_name"
|
||||
# identity.form.password="password"
|
||||
# identity.form.submit="login"
|
||||
|
||||
# What sources should the identity provider consider when determining the
|
||||
# identity associated with a request? Comma separated list of identity sources.
|
||||
# Valid sources: form, visit, http_auth
|
||||
# identity.source="form,http_auth,visit"
|
||||
|
||||
# SqlObjectIdentityProvider
|
||||
# Configuration options for the default IdentityProvider
|
||||
# -------------------------
|
||||
|
||||
# The classes you wish to use for your Identity model. Remember to not use reserved
|
||||
# SQL keywords for class names (at least unless you specify a different table
|
||||
# name using sqlmeta).
|
||||
identity.soprovider.model.user="oilarchive.model.User"
|
||||
identity.soprovider.model.group="oilarchive.model.Group"
|
||||
identity.soprovider.model.permission="oilarchive.model.Permission"
|
||||
|
||||
# The password encryption algorithm used when comparing passwords against what's
|
||||
# stored in the database. Valid values are 'md5' or 'sha1'. If you do not
|
||||
# specify an encryption algorithm, passwords are expected to be clear text.
|
||||
#
|
||||
# The SqlObjectProvider *will* encrypt passwords supplied as part of your login
|
||||
# form. If you set the password through the password property, like:
|
||||
# my_user.password = 'secret'
|
||||
# the password will be encrypted in the database, provided identity is up and
|
||||
# running, or you have loaded the configuration specifying what encryption to
|
||||
# use (in situations where identity may not yet be running, like tests).
|
||||
|
||||
# identity.soprovider.encryption_algorithm=None
|
||||
|
||||
# compress the data sends to the web browser
|
||||
# [/]
|
||||
# gzip_filter.on = True
|
||||
# gzip_filter.mime_types = ["application/x-javascript", "text/javascript", "text/html", "text/css", "text/plain"]
|
||||
|
||||
[/static]
|
||||
static_filter.on = True
|
||||
static_filter.dir = "%(top_level_dir)s/static"
|
||||
|
||||
[/favicon.ico]
|
||||
static_filter.on = True
|
||||
static_filter.file = "%(top_level_dir)s/static/images/favicon.ico"
|
29
oilarchive/config/log.cfg
Normal file
29
oilarchive/config/log.cfg
Normal file
|
@ -0,0 +1,29 @@
|
|||
# LOGGING
|
||||
# Logging is often deployment specific, but some handlers and
|
||||
# formatters can be defined here.
|
||||
|
||||
[logging]
|
||||
[[formatters]]
|
||||
[[[message_only]]]
|
||||
format='*(message)s'
|
||||
|
||||
[[[full_content]]]
|
||||
format='*(asctime)s *(name)s *(levelname)s *(message)s'
|
||||
|
||||
[[handlers]]
|
||||
[[[debug_out]]]
|
||||
class='StreamHandler'
|
||||
level='DEBUG'
|
||||
args='(sys.stdout,)'
|
||||
formatter='full_content'
|
||||
|
||||
[[[access_out]]]
|
||||
class='StreamHandler'
|
||||
level='INFO'
|
||||
args='(sys.stdout,)'
|
||||
formatter='message_only'
|
||||
|
||||
[[[error_out]]]
|
||||
class='StreamHandler'
|
||||
level='ERROR'
|
||||
args='(sys.stdout,)'
|
46
oilarchive/controllers.py
Normal file
46
oilarchive/controllers.py
Normal file
|
@ -0,0 +1,46 @@
|
|||
from turbogears import controllers, expose
|
||||
# from model import *
|
||||
from turbogears import identity, redirect
|
||||
from cherrypy import request, response
|
||||
# from oilarchive import json
|
||||
# import logging
|
||||
# log = logging.getLogger("oilarchive.controllers")
|
||||
|
||||
class Root(controllers.RootController):
|
||||
@expose(template="oilarchive.templates.welcome")
|
||||
# @identity.require(identity.in_group("admin"))
|
||||
def index(self):
|
||||
import time
|
||||
# log.debug("Happy TurboGears Controller Responding For Duty")
|
||||
return dict(now=time.ctime())
|
||||
|
||||
@expose(template="oilarchive.templates.login")
|
||||
def login(self, forward_url=None, previous_url=None, *args, **kw):
|
||||
|
||||
if not identity.current.anonymous \
|
||||
and identity.was_login_attempted() \
|
||||
and not identity.get_identity_errors():
|
||||
raise redirect(forward_url)
|
||||
|
||||
forward_url=None
|
||||
previous_url= request.path
|
||||
|
||||
if identity.was_login_attempted():
|
||||
msg=_("The credentials you supplied were not correct or "
|
||||
"did not grant access to this resource.")
|
||||
elif identity.get_identity_errors():
|
||||
msg=_("You must provide your credentials before accessing "
|
||||
"this resource.")
|
||||
else:
|
||||
msg=_("Please log in.")
|
||||
forward_url= request.headers.get("Referer", "/")
|
||||
|
||||
response.status=403
|
||||
return dict(message=msg, previous_url=previous_url, logging_in=True,
|
||||
original_parameters=request.params,
|
||||
forward_url=forward_url)
|
||||
|
||||
@expose()
|
||||
def logout(self):
|
||||
identity.current.logout()
|
||||
raise redirect("/")
|
35
oilarchive/json.py
Normal file
35
oilarchive/json.py
Normal file
|
@ -0,0 +1,35 @@
|
|||
# A JSON-based API(view) for your app.
|
||||
# Most rules would look like:
|
||||
#
|
||||
# @jsonify.when("isinstance(obj, YourClass)")
|
||||
# def jsonify_yourclass(obj):
|
||||
# return [obj.val1, obj.val2]
|
||||
#
|
||||
# @jsonify can convert your objects to following types:
|
||||
# lists, dicts, numbers and strings
|
||||
|
||||
from turbojson.jsonify import jsonify
|
||||
|
||||
from turbojson.jsonify import jsonify_sqlobject
|
||||
from oilarchive.model import User, Group, Permission
|
||||
|
||||
@jsonify.when('isinstance(obj, Group)')
|
||||
def jsonify_group(obj):
|
||||
result = jsonify_sqlobject( obj )
|
||||
result["users"] = [u.user_name for u in obj.users]
|
||||
result["permissions"] = [p.permission_name for p in obj.permissions]
|
||||
return result
|
||||
|
||||
@jsonify.when('isinstance(obj, User)')
|
||||
def jsonify_user(obj):
|
||||
result = jsonify_sqlobject( obj )
|
||||
del result['password']
|
||||
result["groups"] = [g.group_name for g in obj.groups]
|
||||
result["permissions"] = [p.permission_name for p in obj.permissions]
|
||||
return result
|
||||
|
||||
@jsonify.when('isinstance(obj, Permission)')
|
||||
def jsonify_permission(obj):
|
||||
result = jsonify_sqlobject( obj )
|
||||
result["groups"] = [g.group_name for g in obj.groups]
|
||||
return result
|
105
oilarchive/model.py
Normal file
105
oilarchive/model.py
Normal file
|
@ -0,0 +1,105 @@
|
|||
from datetime import datetime
|
||||
from turbogears.database import PackageHub
|
||||
from sqlobject import *
|
||||
from turbogears import identity
|
||||
|
||||
hub = PackageHub("oilarchive")
|
||||
__connection__ = hub
|
||||
|
||||
# identity models.
|
||||
class Visit(SQLObject):
|
||||
class sqlmeta:
|
||||
table = "visit"
|
||||
|
||||
visit_key = StringCol(length=40, alternateID=True,
|
||||
alternateMethodName="by_visit_key")
|
||||
created = DateTimeCol(default=datetime.now)
|
||||
expiry = DateTimeCol()
|
||||
|
||||
def lookup_visit(cls, visit_key):
|
||||
try:
|
||||
return cls.by_visit_key(visit_key)
|
||||
except SQLObjectNotFound:
|
||||
return None
|
||||
lookup_visit = classmethod(lookup_visit)
|
||||
|
||||
class VisitIdentity(SQLObject):
|
||||
visit_key = StringCol(length=40, alternateID=True,
|
||||
alternateMethodName="by_visit_key")
|
||||
user_id = IntCol()
|
||||
|
||||
|
||||
class Group(SQLObject):
|
||||
"""
|
||||
An ultra-simple group definition.
|
||||
"""
|
||||
|
||||
# names like "Group", "Order" and "User" are reserved words in SQL
|
||||
# so we set the name to something safe for SQL
|
||||
class sqlmeta:
|
||||
table = "tg_group"
|
||||
|
||||
group_name = UnicodeCol(length=16, alternateID=True,
|
||||
alternateMethodName="by_group_name")
|
||||
display_name = UnicodeCol(length=255)
|
||||
created = DateTimeCol(default=datetime.now)
|
||||
|
||||
# collection of all users belonging to this group
|
||||
users = RelatedJoin("User", intermediateTable="user_group",
|
||||
joinColumn="group_id", otherColumn="user_id")
|
||||
|
||||
# collection of all permissions for this group
|
||||
permissions = RelatedJoin("Permission", joinColumn="group_id",
|
||||
intermediateTable="group_permission",
|
||||
otherColumn="permission_id")
|
||||
|
||||
|
||||
class User(SQLObject):
|
||||
"""
|
||||
Reasonably basic User definition. Probably would want additional attributes.
|
||||
"""
|
||||
# names like "Group", "Order" and "User" are reserved words in SQL
|
||||
# so we set the name to something safe for SQL
|
||||
class sqlmeta:
|
||||
table = "tg_user"
|
||||
|
||||
user_name = UnicodeCol(length=16, alternateID=True,
|
||||
alternateMethodName="by_user_name")
|
||||
email_address = UnicodeCol(length=255, alternateID=True,
|
||||
alternateMethodName="by_email_address")
|
||||
display_name = UnicodeCol(length=255)
|
||||
password = UnicodeCol(length=40)
|
||||
created = DateTimeCol(default=datetime.now)
|
||||
|
||||
# groups this user belongs to
|
||||
groups = RelatedJoin("Group", intermediateTable="user_group",
|
||||
joinColumn="user_id", otherColumn="group_id")
|
||||
|
||||
def _get_permissions(self):
|
||||
perms = set()
|
||||
for g in self.groups:
|
||||
perms = perms | set(g.permissions)
|
||||
return perms
|
||||
|
||||
def _set_password(self, cleartext_password):
|
||||
"Runs cleartext_password through the hash algorithm before saving."
|
||||
hash = identity.encrypt_password(cleartext_password)
|
||||
self._SO_set_password(hash)
|
||||
|
||||
def set_password_raw(self, password):
|
||||
"Saves the password as-is to the database."
|
||||
self._SO_set_password(password)
|
||||
|
||||
|
||||
|
||||
class Permission(SQLObject):
|
||||
permission_name = UnicodeCol(length=16, alternateID=True,
|
||||
alternateMethodName="by_permission_name")
|
||||
description = UnicodeCol(length=255)
|
||||
|
||||
groups = RelatedJoin("Group",
|
||||
intermediateTable="group_permission",
|
||||
joinColumn="permission_id",
|
||||
otherColumn="group_id")
|
||||
|
||||
|
14
oilarchive/release.py
Normal file
14
oilarchive/release.py
Normal file
|
@ -0,0 +1,14 @@
|
|||
# Release information about oilarchive
|
||||
|
||||
version = "1.0"
|
||||
|
||||
# description = "Your plan to rule the world"
|
||||
# long_description = "More description about your plan"
|
||||
# author = "Your Name Here"
|
||||
# email = "YourEmail@YourDomain"
|
||||
# copyright = "Vintage 2006 - a good year indeed"
|
||||
|
||||
# if it's open source, you might want to specify these
|
||||
# url = "http://yourcool.site/"
|
||||
# download_url = "http://yourcool.site/download"
|
||||
# license = "MIT"
|
124
oilarchive/static/css/style.css
Normal file
124
oilarchive/static/css/style.css
Normal file
|
@ -0,0 +1,124 @@
|
|||
/*
|
||||
* Quick mash-up of CSS for the TG quick start page.
|
||||
*/
|
||||
|
||||
html, body, th, td {
|
||||
color: black;
|
||||
background-color: #ddd;
|
||||
font: x-small "Lucida Grande", "Lucida Sans Unicode", geneva, verdana, sans-serif;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
#header {
|
||||
height: 80px;
|
||||
width: 777px;
|
||||
background: blue URL('../images/header_inner.png') no-repeat;
|
||||
border-left: 1px solid #aaa;
|
||||
border-right: 1px solid #aaa;
|
||||
margin: 0 auto 0 auto;
|
||||
}
|
||||
|
||||
a.link, a, a.active {
|
||||
color: #369;
|
||||
}
|
||||
|
||||
|
||||
#main_content {
|
||||
color: black;
|
||||
font-size: 127%;
|
||||
background-color: white;
|
||||
width: 757px;
|
||||
margin: 0 auto 0 auto;
|
||||
border-left: 1px solid #aaa;
|
||||
border-right: 1px solid #aaa;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
#sidebar {
|
||||
border: 1px solid #aaa;
|
||||
background-color: #eee;
|
||||
margin: 0.5em;
|
||||
padding: 1em;
|
||||
float: right;
|
||||
width: 200px;
|
||||
font-size: 88%;
|
||||
}
|
||||
|
||||
#sidebar h2 {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
#sidebar ul {
|
||||
margin-left: 1.5em;
|
||||
padding-left: 0;
|
||||
}
|
||||
|
||||
h1,h2,h3,h4,h5,h6,#getting_started_steps {
|
||||
font-family: "Century Schoolbook L", Georgia, serif;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
h2 {
|
||||
font-size: 150%;
|
||||
}
|
||||
|
||||
#getting_started_steps a {
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
#getting_started_steps a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
#getting_started_steps li {
|
||||
font-size: 80%;
|
||||
margin-bottom: 0.5em;
|
||||
}
|
||||
|
||||
#getting_started_steps h2 {
|
||||
font-size: 120%;
|
||||
}
|
||||
|
||||
#getting_started_steps p {
|
||||
font: 100% "Lucida Grande", "Lucida Sans Unicode", geneva, verdana, sans-serif;
|
||||
}
|
||||
|
||||
#footer {
|
||||
border: 1px solid #aaa;
|
||||
border-top: 0px none;
|
||||
color: #999;
|
||||
background-color: white;
|
||||
padding: 10px;
|
||||
font-size: 80%;
|
||||
text-align: center;
|
||||
width: 757px;
|
||||
margin: 0 auto 1em auto;
|
||||
}
|
||||
|
||||
.code {
|
||||
font-family: monospace;
|
||||
}
|
||||
|
||||
span.code {
|
||||
font-weight: bold;
|
||||
background: #eee;
|
||||
}
|
||||
|
||||
#status_block {
|
||||
margin: 0 auto 0.5em auto;
|
||||
padding: 15px 10px 15px 55px;
|
||||
background: #cec URL('../images/ok.png') left center no-repeat;
|
||||
border: 1px solid #9c9;
|
||||
width: 450px;
|
||||
font-size: 120%;
|
||||
font-weight: bolder;
|
||||
}
|
||||
|
||||
.notice {
|
||||
margin: 0.5em auto 0.5em auto;
|
||||
padding: 15px 10px 15px 55px;
|
||||
width: 450px;
|
||||
background: #eef URL('../images/info.png') left center no-repeat;
|
||||
border: 1px solid #cce;
|
||||
}
|
BIN
oilarchive/static/images/favicon.ico
Normal file
BIN
oilarchive/static/images/favicon.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.1 KiB |
BIN
oilarchive/static/images/header_inner.png
Normal file
BIN
oilarchive/static/images/header_inner.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 37 KiB |
BIN
oilarchive/static/images/info.png
Normal file
BIN
oilarchive/static/images/info.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.8 KiB |
BIN
oilarchive/static/images/ok.png
Normal file
BIN
oilarchive/static/images/ok.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 25 KiB |
BIN
oilarchive/static/images/tg_under_the_hood.png
Normal file
BIN
oilarchive/static/images/tg_under_the_hood.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.9 KiB |
BIN
oilarchive/static/images/under_the_hood_blue.png
Normal file
BIN
oilarchive/static/images/under_the_hood_blue.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.6 KiB |
0
oilarchive/templates/__init__.py
Normal file
0
oilarchive/templates/__init__.py
Normal file
112
oilarchive/templates/login.kid
Normal file
112
oilarchive/templates/login.kid
Normal file
|
@ -0,0 +1,112 @@
|
|||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
|
||||
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml"
|
||||
xmlns:py="http://purl.org/kid/ns#">
|
||||
|
||||
<head>
|
||||
<meta content="text/html; charset=UTF-8"
|
||||
http-equiv="content-type" py:replace="''"/>
|
||||
<title>Login</title>
|
||||
<style type="text/css">
|
||||
#loginBox
|
||||
{
|
||||
width: 30%;
|
||||
margin: auto;
|
||||
margin-top: 10%;
|
||||
padding-left: 10%;
|
||||
padding-right: 10%;
|
||||
padding-top: 5%;
|
||||
padding-bottom: 5%;
|
||||
font-family: verdana;
|
||||
font-size: 10px;
|
||||
background-color: #eee;
|
||||
border: 2px solid #ccc;
|
||||
}
|
||||
|
||||
#loginBox h1
|
||||
{
|
||||
font-size: 42px;
|
||||
font-family: "Trebuchet MS";
|
||||
margin: 0;
|
||||
color: #ddd;
|
||||
}
|
||||
|
||||
#loginBox p
|
||||
{
|
||||
position: relative;
|
||||
top: -1.5em;
|
||||
padding-left: 4em;
|
||||
font-size: 12px;
|
||||
margin: 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
#loginBox table
|
||||
{
|
||||
table-layout: fixed;
|
||||
border-spacing: 0;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
#loginBox td.label
|
||||
{
|
||||
width: 33%;
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
#loginBox td.field
|
||||
{
|
||||
width: 66%;
|
||||
}
|
||||
|
||||
#loginBox td.field input
|
||||
{
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
#loginBox td.buttons
|
||||
{
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div id="loginBox">
|
||||
<h1>Login</h1>
|
||||
<p>${message}</p>
|
||||
<form action="${previous_url}" method="POST">
|
||||
<table>
|
||||
<tr>
|
||||
<td class="label">
|
||||
<label for="user_name">User Name:</label>
|
||||
</td>
|
||||
<td class="field">
|
||||
<input type="text" id="user_name" name="user_name"/>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td class="label">
|
||||
<label for="password">Password:</label>
|
||||
</td>
|
||||
<td class="field">
|
||||
<input type="password" id="password" name="password"/>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td colspan="2" class="buttons">
|
||||
<input type="submit" name="login" value="Login"/>
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
<input py:if="forward_url" type="hidden" name="forward_url"
|
||||
value="${forward_url}"/>
|
||||
|
||||
<input py:for="name,value in original_parameters.items()"
|
||||
type="hidden" name="${name}" value="${value}"/>
|
||||
</form>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
48
oilarchive/templates/master.kid
Normal file
48
oilarchive/templates/master.kid
Normal file
|
@ -0,0 +1,48 @@
|
|||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<?python import sitetemplate ?>
|
||||
<html xmlns="http://www.w3.org/1999/xhtml" xmlns:py="http://purl.org/kid/ns#" py:extends="sitetemplate">
|
||||
|
||||
<head py:match="item.tag=='{http://www.w3.org/1999/xhtml}head'" py:attrs="item.items()">
|
||||
<meta content="text/html; charset=UTF-8" http-equiv="content-type" py:replace="''"/>
|
||||
<title py:replace="''">Your title goes here</title>
|
||||
<meta py:replace="item[:]"/>
|
||||
<style type="text/css">
|
||||
#pageLogin
|
||||
{
|
||||
font-size: 10px;
|
||||
font-family: verdana;
|
||||
text-align: right;
|
||||
}
|
||||
</style>
|
||||
<style type="text/css" media="screen">
|
||||
@import "/static/css/style.css";
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body py:match="item.tag=='{http://www.w3.org/1999/xhtml}body'" py:attrs="item.items()">
|
||||
<div py:if="tg.config('identity.on',False) and not 'logging_in' in locals()"
|
||||
id="pageLogin">
|
||||
<span py:if="tg.identity.anonymous">
|
||||
<a href="/login">Login</a>
|
||||
</span>
|
||||
<span py:if="not tg.identity.anonymous">
|
||||
Welcome ${tg.identity.user.display_name}.
|
||||
<a href="/logout">Logout</a>
|
||||
</span>
|
||||
</div>
|
||||
<div id="header"> </div>
|
||||
<div id="main_content">
|
||||
<div py:if="tg_flash" class="flash" py:content="tg_flash"></div>
|
||||
|
||||
<div py:replace="[item.text]+item[:]"/>
|
||||
|
||||
<!-- End of main_content -->
|
||||
</div>
|
||||
<div id="footer"> <img src="/static/images/under_the_hood_blue.png" alt="TurboGears under the hood" />
|
||||
<p>TurboGears is a open source front-to-back web development
|
||||
framework written in Python</p>
|
||||
<p>Copyright © 2006 Kevin Dangoor</p>
|
||||
</div>
|
||||
</body>
|
||||
|
||||
</html>
|
48
oilarchive/templates/welcome.kid
Normal file
48
oilarchive/templates/welcome.kid
Normal file
|
@ -0,0 +1,48 @@
|
|||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml" xmlns:py="http://purl.org/kid/ns#"
|
||||
py:extends="'master.kid'">
|
||||
<head>
|
||||
<meta content="text/html; charset=utf-8" http-equiv="Content-Type" py:replace="''"/>
|
||||
<title>Welcome to TurboGears</title>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<div id="status_block">Your application is now running</div>
|
||||
<div id="sidebar">
|
||||
<h2>Learn more</h2>
|
||||
Learn more about TurboGears and take part in its
|
||||
development
|
||||
<ul class="links">
|
||||
<li><a href="http://www.turbogears.org">Official website</a></li>
|
||||
<li><a href="http://docs.turbogears.org">Documentation</a></li>
|
||||
<li><a href="http://trac.turbogears.org/turbogears/">Trac
|
||||
(bugs/suggestions)</a></li>
|
||||
<li><a href="http://groups.google.com/group/turbogears"> Mailing list</a> </li>
|
||||
</ul>
|
||||
<span py:replace="now">now</span>
|
||||
</div>
|
||||
<div id="getting_started">
|
||||
<ol id="getting_started_steps">
|
||||
<li class="getting_started">
|
||||
<h3>Model</h3>
|
||||
<p> <a href="http://docs.turbogears.org/1.0/GettingStarted/DefineDatabase">Design models</a> in the <span class="code">model.py</span>.<br/>
|
||||
Edit <span class="code">dev.cfg</span> to <a href="http://docs.turbogears.org/1.0/GettingStarted/UseDatabase">use a different backend</a>, or start with a pre-configured SQLite database. <br/>
|
||||
Use script <span class="code">tg-admin sql create</span> to create the database tables.</p>
|
||||
</li>
|
||||
<li class="getting_started">
|
||||
<h3>View</h3>
|
||||
<p> Edit <a href="http://docs.turbogears.org/1.0/GettingStarted/Kid">html-like templates</a> in the <span class="code">/templates</span> folder;<br/>
|
||||
Put all <a href="http://docs.turbogears.org/1.0/StaticFiles">static contents</a> in the <span class="code">/static</span> folder. </p>
|
||||
</li>
|
||||
<li class="getting_started">
|
||||
<h3>Controller</h3>
|
||||
<p> Edit <span class="code"> controllers.py</span> and <a href="http://docs.turbogears.org/1.0/GettingStarted/CherryPy">build your
|
||||
website structure</a> with the simplicity of Python objects. <br/>
|
||||
TurboGears will automatically reload itself when you modify your project. </p>
|
||||
</li>
|
||||
</ol>
|
||||
<div class="notice"> If you create something cool, please <a href="http://groups.google.com/group/turbogears">let people know</a>, and consider contributing something back to the <a href="http://groups.google.com/group/turbogears">community</a>.</div>
|
||||
</div>
|
||||
<!-- End of getting_started -->
|
||||
</body>
|
||||
</html>
|
0
oilarchive/tests/__init__.py
Normal file
0
oilarchive/tests/__init__.py
Normal file
33
oilarchive/tests/test_controllers.py
Normal file
33
oilarchive/tests/test_controllers.py
Normal file
|
@ -0,0 +1,33 @@
|
|||
import turbogears
|
||||
from nose import with_setup
|
||||
from turbogears import testutil
|
||||
from oilarchive.controllers import Root
|
||||
import cherrypy
|
||||
|
||||
def teardown_func():
|
||||
"""Tests for apps using identity need to stop CP/TG after each test to
|
||||
stop the VisitManager thread. See http://trac.turbogears.org/turbogears/ticket/1217
|
||||
for details.
|
||||
"""
|
||||
turbogears.startup.stopTurboGears()
|
||||
|
||||
cherrypy.root = Root()
|
||||
|
||||
def test_method():
|
||||
"the index method should return a string called now"
|
||||
import types
|
||||
result = testutil.call(cherrypy.root.index)
|
||||
assert type(result["now"]) == types.StringType
|
||||
test_method = with_setup(teardown=teardown_func)(test_method)
|
||||
|
||||
def test_indextitle():
|
||||
"The indexpage should have the right title"
|
||||
testutil.createRequest("/")
|
||||
assert "<TITLE>Welcome to TurboGears</TITLE>" in cherrypy.response.body[0]
|
||||
test_indextitle = with_setup(teardown=teardown_func)(test_indextitle)
|
||||
|
||||
def test_logintitle():
|
||||
"login page should have the right title"
|
||||
testutil.createRequest("/login")
|
||||
assert "<TITLE>Login</TITLE>" in cherrypy.response.body[0]
|
||||
test_logintitle = with_setup(teardown=teardown_func)(test_logintitle)
|
23
oilarchive/tests/test_model.py
Normal file
23
oilarchive/tests/test_model.py
Normal file
|
@ -0,0 +1,23 @@
|
|||
# If your project uses a database, you can set up database tests
|
||||
# similar to what you see below. Be sure to set the db_uri to
|
||||
# an appropriate uri for your testing database. sqlite is a good
|
||||
# choice for testing, because you can use an in-memory database
|
||||
# which is very fast.
|
||||
|
||||
from turbogears import testutil, database
|
||||
# from oilarchive.model import YourDataClass, User
|
||||
|
||||
# database.set_db_uri("sqlite:///:memory:")
|
||||
|
||||
# class TestUser(testutil.DBTest):
|
||||
# def get_model(self):
|
||||
# return User
|
||||
#
|
||||
# def test_creation(self):
|
||||
# "Object creation should set the name"
|
||||
# obj = User(user_name = "creosote",
|
||||
# email_address = "spam@python.not",
|
||||
# display_name = "Mr Creosote",
|
||||
# password = "Wafer-thin Mint")
|
||||
# assert obj.display_name == "Mr Creosote"
|
||||
|
78
sample-prod.cfg
Normal file
78
sample-prod.cfg
Normal file
|
@ -0,0 +1,78 @@
|
|||
[global]
|
||||
# This is where all of your settings go for your production environment.
|
||||
# You'll copy this file over to your production server and provide it
|
||||
# as a command-line option to your start script.
|
||||
# Settings that are the same for both development and production
|
||||
# (such as template engine, encodings, etc.) all go in
|
||||
# oilarchive/config/app.cfg
|
||||
|
||||
# DATABASE
|
||||
|
||||
# pick the form for your database
|
||||
# sqlobject.dburi="postgres://username@hostname/databasename"
|
||||
# sqlobject.dburi="mysql://username:password@hostname:port/databasename"
|
||||
# sqlobject.dburi="sqlite:///file_name_and_path"
|
||||
|
||||
# If you have sqlite, here's a simple default to get you started
|
||||
# in development
|
||||
sqlobject.dburi="sqlite://%(current_dir_uri)s/devdata.sqlite"
|
||||
|
||||
|
||||
# if you are using a database or table type without transactions
|
||||
# (MySQL default, for example), you should turn off transactions
|
||||
# by prepending notrans_ on the uri
|
||||
# sqlobject.dburi="notrans_mysql://username:password@hostname:port/databasename"
|
||||
|
||||
# for Windows users, sqlite URIs look like:
|
||||
# sqlobject.dburi="sqlite:///drive_letter:/path/to/file"
|
||||
|
||||
|
||||
# SERVER
|
||||
|
||||
server.environment="production"
|
||||
|
||||
# Sets the number of threads the server uses
|
||||
# server.thread_pool = 1
|
||||
|
||||
# if this is part of a larger site, you can set the path
|
||||
# to the TurboGears instance here
|
||||
# server.webpath=""
|
||||
|
||||
# session_filter.on = True
|
||||
|
||||
# Set to True if you'd like to abort execution if a controller gets an
|
||||
# unexpected parameter. False by default
|
||||
# tg.strict_parameters = False
|
||||
|
||||
# Set the following to True if you are deploying your app using mod_proxy,
|
||||
# mod_rewrite or any other mechanism that forwards requests to your app.
|
||||
# base_url_filter.on = False
|
||||
# base_url_filter.use_x_forwarded_host = False
|
||||
|
||||
# LOGGING
|
||||
# Logging configuration generally follows the style of the standard
|
||||
# Python logging module configuration. Note that when specifying
|
||||
# log format messages, you need to use *() for formatting variables.
|
||||
# Deployment independent log configuration is in oilarchive/config/log.cfg
|
||||
[logging]
|
||||
|
||||
[[handlers]]
|
||||
|
||||
[[[access_out]]]
|
||||
# set the filename as the first argument below
|
||||
args="('server.log',)"
|
||||
class='FileHandler'
|
||||
level='INFO'
|
||||
formatter='message_only'
|
||||
|
||||
[[loggers]]
|
||||
[[[oilarchive]]]
|
||||
level='ERROR'
|
||||
qualname='oilarchive'
|
||||
handlers=['error_out']
|
||||
|
||||
[[[access]]]
|
||||
level='INFO'
|
||||
qualname='turbogears.access'
|
||||
handlers=['access_out']
|
||||
propagate=0
|
62
setup.py
Normal file
62
setup.py
Normal file
|
@ -0,0 +1,62 @@
|
|||
from setuptools import setup, find_packages
|
||||
from turbogears.finddata import find_package_data
|
||||
|
||||
import os
|
||||
execfile(os.path.join("oilarchive", "release.py"))
|
||||
|
||||
setup(
|
||||
name="oilarchive",
|
||||
version=version,
|
||||
|
||||
# uncomment the following lines if you fill them out in release.py
|
||||
#description=description,
|
||||
#author=author,
|
||||
#author_email=email,
|
||||
#url=url,
|
||||
#download_url=download_url,
|
||||
#license=license,
|
||||
|
||||
install_requires = [
|
||||
"TurboGears >= 1.0",
|
||||
],
|
||||
scripts = ["start-oilarchive.py"],
|
||||
zip_safe=False,
|
||||
packages=find_packages(),
|
||||
package_data = find_package_data(where='oilarchive',
|
||||
package='oilarchive'),
|
||||
keywords = [
|
||||
# Use keywords if you'll be adding your package to the
|
||||
# Python Cheeseshop
|
||||
|
||||
# if this has widgets, uncomment the next line
|
||||
# 'turbogears.widgets',
|
||||
|
||||
# if this has a tg-admin command, uncomment the next line
|
||||
# 'turbogears.command',
|
||||
|
||||
# if this has identity providers, uncomment the next line
|
||||
# 'turbogears.identity.provider',
|
||||
|
||||
# If this is a template plugin, uncomment the next line
|
||||
# 'python.templating.engines',
|
||||
|
||||
# If this is a full application, uncomment the next line
|
||||
# 'turbogears.app',
|
||||
],
|
||||
classifiers = [
|
||||
'Development Status :: 3 - Alpha',
|
||||
'Operating System :: OS Independent',
|
||||
'Programming Language :: Python',
|
||||
'Topic :: Software Development :: Libraries :: Python Modules',
|
||||
'Framework :: TurboGears',
|
||||
# if this is an application that you'll distribute through
|
||||
# the Cheeseshop, uncomment the next line
|
||||
# 'Framework :: TurboGears :: Applications',
|
||||
|
||||
# if this is a package that includes widgets that you'll distribute
|
||||
# through the Cheeseshop, uncomment the next line
|
||||
# 'Framework :: TurboGears :: Widgets',
|
||||
],
|
||||
test_suite = 'nose.collector',
|
||||
)
|
||||
|
BIN
setup.pyc
Normal file
BIN
setup.pyc
Normal file
Binary file not shown.
25
start-oilarchive.py
Executable file
25
start-oilarchive.py
Executable file
|
@ -0,0 +1,25 @@
|
|||
#!/Library/Frameworks/Python.framework/Versions/2.4/Resources/Python.app/Contents/MacOS/Python
|
||||
import pkg_resources
|
||||
pkg_resources.require("TurboGears")
|
||||
|
||||
from turbogears import update_config, start_server
|
||||
import cherrypy
|
||||
cherrypy.lowercase_api = True
|
||||
from os.path import *
|
||||
import sys
|
||||
|
||||
# first look on the command line for a desired config file,
|
||||
# if it's not on the command line, then
|
||||
# look for setup.py in this directory. If it's not there, this script is
|
||||
# probably installed
|
||||
if len(sys.argv) > 1:
|
||||
update_config(configfile=sys.argv[1],
|
||||
modulename="oilarchive.config")
|
||||
elif exists(join(dirname(__file__), "setup.py")):
|
||||
update_config(configfile="dev.cfg",modulename="oilarchive.config")
|
||||
else:
|
||||
update_config(configfile="prod.cfg",modulename="oilarchive.config")
|
||||
|
||||
from oilarchive.controllers import Root
|
||||
|
||||
start_server(Root())
|
5
test.cfg
Normal file
5
test.cfg
Normal file
|
@ -0,0 +1,5 @@
|
|||
# You can place test-specific configuration options here (like test db uri, etc)
|
||||
#
|
||||
|
||||
sqlobject.dburi = "sqlite:///:memory:"
|
||||
|
Loading…
Reference in a new issue