Only fools and nerds create their own CMS nowadays. Hurray! Anyway, so
here it is, w20e.pycms. Using the Pyramid framework as its base,
building on top of good old Zope (and some Plone) concepts. The CMS is
created using these main concepts:
* ZODB as database
* ZCML as configuration/glue language
* small core
* optional components (like search, catalog, sharing); just include what
The CMS is a framework, not an out of the box app. What you'll need to
do is create your own Pyramid app, using the CMS as base. We've tried
to make this as easy as possible: use w20e.pycms.sitemaker (to be
found on github or pypi) to obtain a paster template for your app. Run
paster to create your app, and there you go...
w20e.pycms is Yet Another CMS, but without using the acronym. Why,
for the flying spaghetti monsters sake another CMS? Well, you know how
it goes. You use Plone for some years, find out that when your
favourite tool is a hammer, all problems have a rather strong tendency
towards nailishness... Then Pyramid comes along, giving you the best
of Plone (ZODB, Zope Component Architecture, ZCML, Chameleon, etc.)
for creating lightweight apps. Then you need a Page with
WYSIWYG... then you need search... sharing. Then you wake up with a
basic CMS in your hands. May as well share it so you can decide for
yourself whether it is worth your while. I mean, you dont __have__ to
Anyway, read on if you like...
w20e.pycms is not for the faint of heart, nor for people that cannot
read Python code, hate programming, think that the use of XML for
configuration is sooo 1990, are convinced that Windows 95 was the best
OS ever or would preferrably use a rocket launcher to deal with vermin
in the kitchen. It _is_ on the other hand, for those that rank
fuckit.js among the best JS libraries ever, enjoy Terry Pratchett,
love buildout en ZCML and think that beer is so much more that just a
Our little CMS gives you a framework to build your sites upon, if
you're not targeting the enterprise market. If you do, be gone (to the
PyCMS gives you:
* ZODB for storing data
* user & group management
* search, using repoze.catalog (optional)
* creating and maintaining pages
* an easily extendable framework for new content types
* a lot of ZCML configuration
* CMS design based on (Twitter) Bootstrap
We assume that you know how to use buildout, create virtual
environments, like to use paster, etc. But this is only one way to get
First, create a package for your project, requiring:
The easiest way to do so, is using our paster template
pycms_project. Install the w20e.pycms.sitemaker package (get it from
github), something along these lines:
# virtualenv <env>
# cd <env>; ./bin/activate
# ./bin/easy_install w20e.pycms.sitemaker
# cd <wherever you'd="" like="" your="" app="" sources="">
# paster create -t pycms_project <package name="">
If you really want to do it by hand, create an __init__ file for your
Pyramid app like this:
from w20e.pycms import make_pycms_app
def main(global_config, **settings):
return make_pycms_app(__package__, **settings)
and Bob might be your Uncle.
Secondly, create a buildout and virtualenv for your stuff. Why not use
w20e.buildoutskel? Install it using easy_install, and
# cd <whereever you="" want="" your="" buildout="" files="">
# paster create -t buildout
and answer <package name=""> to the project name question, and pycms to
the type question.
You now have a bunch of buildout files, almost ready to run your app!
You most likely will consider creating a buildout-my.cfg that extends
buildout-base.cfg, and adds some develop paths, like:
<that path="" where="" your="" pycms="" app="" was="" created,="" and="" where="" the="" setup.py="" resides...="">
Last, run python bootstrap.py and then buildout with your config file.
Now it's time to rev up the engine, and see what has happend. Run your
app like so (within the buildout dir):
# ./bin/paster serve dvl.ini [--reload]
Direct your favorite browser (most likely Lynx or Mosaic) to
http://localhost:6543/ and sit back and relax!
You may or may not be totally satisfied with the result so far. If
this is utterly your idea of a superduper web app, good on ya! If not,
- Add the default management and public css / js files (if you want):
add this to your configure.zcml:
<include package="w20e.pycms" file="public_resources.zcml"/>
<include package="w20e.pycms" file="manage_resources.zcml"/>
- Include any other CSS and JS you like, using the pycms zcml directives:
- Override assets like favicon and robots.txt:
- Most likely you'll want to override the 'content' macro, that is
called to display a page. To do this, make your own pt file, make
that extend 'main.macros['master'], and let it fill the 'body' slot:
<metal:define-macro define-macro="master" <br=""> metal:extend-macro="main.macros['master']">
<html xmlns="http://www.w3.org/1999/xhtml" <br=""> xml:lang="en"
Good morning Grommit...
and add to your configure.zcml (always assuming you called your macro
Using the CMS: core concepts
The core CMS consists solely of pages. Pages are just things with
text. Nothing serious. You may want to create your own content types,
actions, etc. Luckily that's not hard to do. Best way is to look at
existing code... Anyway, some examples here:
You can register actions with your content. The currently used setup
mainly uses 'perspectives' or ways to look at your content. In the
management interface these are rendered as 'tabs'. Actions are
configured using zcml. Use the action statment as follows:
ctype is an optional filter.
Create your own content types if you wish. You can register an icon,
and possible subtypes with your type using:
Your actual model should extend either
You may want to use w20e.forms (read: should) for your model. You can
either create an xml form that describes your model, or if you insist
on doing things more Pythonic, create your form
programmatically. Checkout out w20e.forms for details.
A simple model looks like this:
from w20e.pycms.models.base import BaseContent
""" Well, actually it's more like an 'object'... """
def __init__(self, content_id, data=None):
BaseContent.__init__(self, content_id, data)
You can configure how your form for editing and adding is
found. Default is that PyCMS looks for a file in <your package<br="">home>/forms/<content type="">.xml, so in this case:
If you want something completely different, configure an adapter for your
content type that provides a form factory:
And make sure it actually implements IFormFactory and can create a
An alternative for defining content types is defining 'natures'. Let's
face it: what is so special about an event? It is really just a page
thing with a start- and end date and a location, isn't it? And what
about news? Isn't that not very much like a page too? If you agree,
A page can be not only a page, but it can also be news-ish, or
event-ish. That is it's nature. You can register natures like so:
This will make the nature show up in the 'natures' dropdown menu. Now
either you leave it like this, or you also modify the form for the
page with an w20e.forms.interfaces.IFormModifier implementation:
And create a class Event along these lines:
from zope.interface import implements
from w20e.forms.interfaces import IForm, IFormModifier
from w20e.forms.data.field import Field
from w20e.forms.model.fieldproperties import FieldProperties
from w20e.forms.rendering.control import Input
from w20e.forms.rendering.group import FlowGroup
def __init__(self, form):
self.form = form
def modify(self, form):
""" Add begin, end and location to form """
grp = FlowGroup("eventgroup", label="Event")
grp.addRenderable(Input("start", "Start of event",
grp.addRenderable(Input("end", "End of event", bind="end"))
grp.addRenderable(Input("location", "Location", bind="location"))
Or whatever you think should be added to the page form...
The default robots.txt allows all. Override as per your liking...
Would you like search enabled for your site?
Add this to your configure.zcml:
<include package="w20e.pycms" file="search.zcml"/>
Would you like search enabled for your site?
Add this to your configure.zcml:
and this to your setup dependencies (don't forget to run buildout):
pycms.acl.force_new = True|False
Force new version of ACL. All your security data will be lost
pycms.catalog.force_new = True|False
Force new version of catalog. All your entries will be lost, but you
can just run reindex-catalog and all is well again...
pycms.admin_user = <user>:<pwd>
Admin user and password, like so: pycms.admin_user = admin:pipo
pycms.admin_secret = <somesecret>
This secret may be used as URL parameter to obtain admin permission
Use it wisely!
pycms.minify_css = True|False
Minify CSS. Defaults to False
pycms.minify_js = True|False
Minify JS. Defaults to False
pycms.logged_in_redirect = <url>
pycms.rootclass = <full dotted="" classname="">
Defaults to w20e.pycms.models.site.Site
pycms.roottitle = <string>
Defaults to "Welcome"
Send email as ...
Send also to bcc
Where to go after successfull add
Where to go after cancelled add
Where to go after delete
pycms.tempregister.timout = <int>
minimal amount of seconds before a temporary object might be removed
* json representation of pycms content types
* upgrade jquery + jqueryui
* small fixes
* initially created objects will have a temporary state, until they
have been completed
* add_macros now within base view class, so it's easier to override
* validation.js fixed for bootstrap class names
* Finally we have a serious (?) licence. Also updated some metadata.
* Enabled call to render viewlet through base view