Skip to content


Subversion checkout URL

You can clone with
Download ZIP
a simple blog engine created with django
Branch: master

Fetching latest commit…

Cannot retrieve the latest commit at this time

Failed to load latest commit information.




the blogengine behind origiNell. My aim is to keep it, as the name suggests, simple, fast and easy to integrate/customize.


and of course


Use the included

python install

Then open up your project's and include this app's urls. For example:

(r'^blog/', include('django_simpleblog.urls.entries')),


You can specify the following options in your

    If True, tags will be enabled. There's a check if django-tagging is installed. If so, then we'll use that as tagfield, otherwise tags will be handled as plain-text. Default: True


    If True, category support will be enabled. Default: True


    If True, this will enable Markdown markup language. Default: True


    A list containing available markdown extensions Default: ['codehilite',]


    If True enables django's comments framework Default: True


    If True enables email notification about new comments Default: False


    If True, a comment containing the <a href=..> html tag, will NOT be displayed on the website. It's is_public attribute will therefore be set to False. Default: True

Note: If you don't want categories you need to set CATEGORIES to False before the initial syncdb. Otherwise you'll need perform the sql changes manually or by using one of the nice db altering utilities for django (django-evolution, South, dmigration,...)

RSS/Atom Feeds

To enable RSS/Atom feeds simply add the following to your

(r'^feeds/', include('django_simpleblog.urls.feeds')),

This will enable two feeds: latest entries and if you set CATEGORIES to true, a category based feed.

Feeds can be customized via the FEED_SETTINGS dictionary in your Here are the default settings. If you didn't activate CATEGORIES, ignore the 'category' part.

FEED_SETTINGS = {'latest': {'title': 'Simpleblog Latest 10 Entries',
                            'link':  '/news/',
                            'description': 'Simpleblog Newsfeed',
                            'length': 10,
                            'item_descr_length': 48,},
                 'category': {'title': 'Simpleblog Categoryfeed',
                              'link': '/category/',
                              'description': 'Feed For Category "%s"'},

Note that %s in category title and description will be replaced by the categoryname, if specified.

Comment Moderation

We now have a simple option to moderate an entry based on it's content. Thanks to django's very flexible comments framework, it's a breeze (see

class EntryModerator(CommentModerator):
    email_notification = COMMENTS_NOTIFICATION
    enable_field = \'enable_comments\'

        def moderate(self, comment, content_object, request):
            if self.auto_moderate_field and self.moderate_after:
                if self._get_delta(, getattr(content_object, self.auto_moderate_field)).days >= self.moderate_after:
                    return True
            if 'a href' in comment.comment:
                return True
            return False
moderator.register(Entry, EntryModerator)

As you can see right now we simply check for 'a href' in a comment. This is the most used html tag I've seen in spam. Anyway, you may have noticed that you can actually have access to the comment object, content_object and request. This of course enables a hell lot of ways to refine spam protection (or you just tie Akismet in at this point).

Note: "if 'a href' in" is not case insensitive. If you want case insensitivity you either have to do lower(comment.comment) or use a regular expression.

A little tip for everyone who wants the user, or rather say spambot, to know that his/her comment has been moderated. I bet you already customised comments/posted.html to fit your need. Notice that you actually have full access to the comment itself inside this template!

Here's what I did:

{% if comment.is_public %}
    <h1>{% trans "Thank you for your comment" %}.</h1><br />
    <a href="{{ comment.get_absolute_url }}">View it!</a>
{% else %}
    <h1>Your comment has been moderated because of denied usage of HTML</h1><br />
    It must be reviewed and approved by the admin.
{% endif %}

If you need more advanced spam protection, I suggest you take a look at akismet integration in django. There are a lot of blog posts about it. In case you want to outsource comments, I highly recommend you take a look at Disqus in combination with django-disqus. I have also blogged about the comment migration.



Creates a humanized enumeration of a specified queryset. Uses typographically correct ambidextrous quotes.


{% human_enum YourQueryset %}

or if you want to have urls to the object:

{% human_enum YourQueryset urlify %}

The __unicode__() method in your model should return some kind of title of your object. If you enable the option to build urls, your model needs a get_absolute_url() method.


Recently he wrote {% human_enum Entry.objects.all() %}.

Gives you:

Recently he wrote “Title 1”, “Title 2”, “Title 3” and “Title 4”.


Returns the newest entry.


{% latest_entry as entry %}


  • How to use syntax highlighting in a post

  • See how to generate a css for pygment's here

  • To get the latest entry you just have to do the following in a template:

    {% load latest %}
    {% latest_entry as [varname] %}
  • Check if an entry has been modified

    {% ifnotequal object.created object.modified %}..{% endifnotequal %}

Something went wrong with that request. Please try again.