Initial commit
This commit is contained in:
commit
cfaa2b2a50
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
/venv/
|
||||||
|
.coverage
|
||||||
|
db.sqlite3
|
||||||
|
/htmlcov/
|
6
CONTRIBUTE.md
Normal file
6
CONTRIBUTE.md
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
If you want to contribute, put your ideas on the issue board; the
|
||||||
|
project leads will mark it as a feature, and discussion can start if
|
||||||
|
it is required or not.
|
||||||
|
|
||||||
|
If you find a bug in the code somewhere, feel free to fix it and issue
|
||||||
|
a pull request against master. The same goes for features.
|
13
COPYING
Normal file
13
COPYING
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Affero General Public License as
|
||||||
|
published by the Free Software Foundation, either version 3 of the
|
||||||
|
License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Affero General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Affero General Public
|
||||||
|
License along with this program. If not, see
|
||||||
|
<http://www.gnu.org/licenses/>.
|
13
README.md
Normal file
13
README.md
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
100(ish) words challenge
|
||||||
|
========================
|
||||||
|
|
||||||
|
This is the main project for 100(ish)word challenge, a challenge for
|
||||||
|
artists of different kinds.
|
||||||
|
|
||||||
|
The goal of the site is to present a word for the users every day, and
|
||||||
|
let them create something based on that word, within a given time
|
||||||
|
limit. They can then share their work via the site, so other users can
|
||||||
|
vote for it (if it represents the word by any means).
|
||||||
|
|
||||||
|
The project aims to run on Heroku, but testing is done for both Python
|
||||||
|
2.7 (because of Heroku) and Python 3.4 (because we can).
|
10
manage.py
Executable file
10
manage.py
Executable file
@ -0,0 +1,10 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "wordchallenge.settings")
|
||||||
|
|
||||||
|
from django.core.management import execute_from_command_line
|
||||||
|
|
||||||
|
execute_from_command_line(sys.argv)
|
0
wordchallenge/__init__.py
Normal file
0
wordchallenge/__init__.py
Normal file
103
wordchallenge/settings.py
Normal file
103
wordchallenge/settings.py
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
"""
|
||||||
|
Django settings for wordchallenge project.
|
||||||
|
|
||||||
|
Generated by 'django-admin startproject' using Django 1.8.7.
|
||||||
|
|
||||||
|
For more information on this file, see
|
||||||
|
https://docs.djangoproject.com/en/1.8/topics/settings/
|
||||||
|
|
||||||
|
For the full list of settings and their values, see
|
||||||
|
https://docs.djangoproject.com/en/1.8/ref/settings/
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
|
||||||
|
import os
|
||||||
|
|
||||||
|
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
||||||
|
|
||||||
|
|
||||||
|
# Quick-start development settings - unsuitable for production
|
||||||
|
# See https://docs.djangoproject.com/en/1.8/howto/deployment/checklist/
|
||||||
|
|
||||||
|
# SECURITY WARNING: keep the secret key used in production secret!
|
||||||
|
SECRET_KEY = '8$#!)33w0dfn#!ttb2p+)0a*i)n1&q6gwecasfic0+^idz4%qk'
|
||||||
|
|
||||||
|
# SECURITY WARNING: don't run with debug turned on in production!
|
||||||
|
DEBUG = True
|
||||||
|
|
||||||
|
ALLOWED_HOSTS = []
|
||||||
|
|
||||||
|
|
||||||
|
# Application definition
|
||||||
|
|
||||||
|
INSTALLED_APPS = (
|
||||||
|
'django.contrib.admin',
|
||||||
|
'django.contrib.auth',
|
||||||
|
'django.contrib.contenttypes',
|
||||||
|
'django.contrib.sessions',
|
||||||
|
'django.contrib.messages',
|
||||||
|
'django.contrib.staticfiles',
|
||||||
|
'words',
|
||||||
|
)
|
||||||
|
|
||||||
|
MIDDLEWARE_CLASSES = (
|
||||||
|
'django.contrib.sessions.middleware.SessionMiddleware',
|
||||||
|
'django.middleware.common.CommonMiddleware',
|
||||||
|
'django.middleware.csrf.CsrfViewMiddleware',
|
||||||
|
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
||||||
|
'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
|
||||||
|
'django.contrib.messages.middleware.MessageMiddleware',
|
||||||
|
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
||||||
|
'django.middleware.security.SecurityMiddleware',
|
||||||
|
)
|
||||||
|
|
||||||
|
ROOT_URLCONF = 'wordchallenge.urls'
|
||||||
|
|
||||||
|
TEMPLATES = [
|
||||||
|
{
|
||||||
|
'BACKEND': 'django.template.backends.django.DjangoTemplates',
|
||||||
|
'DIRS': [],
|
||||||
|
'APP_DIRS': True,
|
||||||
|
'OPTIONS': {
|
||||||
|
'context_processors': [
|
||||||
|
'django.template.context_processors.debug',
|
||||||
|
'django.template.context_processors.request',
|
||||||
|
'django.contrib.auth.context_processors.auth',
|
||||||
|
'django.contrib.messages.context_processors.messages',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
WSGI_APPLICATION = 'wordchallenge.wsgi.application'
|
||||||
|
|
||||||
|
|
||||||
|
# Database
|
||||||
|
# https://docs.djangoproject.com/en/1.8/ref/settings/#databases
|
||||||
|
|
||||||
|
DATABASES = {
|
||||||
|
'default': {
|
||||||
|
'ENGINE': 'django.db.backends.sqlite3',
|
||||||
|
'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# Internationalization
|
||||||
|
# https://docs.djangoproject.com/en/1.8/topics/i18n/
|
||||||
|
|
||||||
|
LANGUAGE_CODE = 'en-us'
|
||||||
|
|
||||||
|
TIME_ZONE = 'UTC'
|
||||||
|
|
||||||
|
USE_I18N = True
|
||||||
|
|
||||||
|
USE_L10N = True
|
||||||
|
|
||||||
|
USE_TZ = True
|
||||||
|
|
||||||
|
|
||||||
|
# Static files (CSS, JavaScript, Images)
|
||||||
|
# https://docs.djangoproject.com/en/1.8/howto/static-files/
|
||||||
|
|
||||||
|
STATIC_URL = '/static/'
|
21
wordchallenge/urls.py
Normal file
21
wordchallenge/urls.py
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
"""wordchallenge URL Configuration
|
||||||
|
|
||||||
|
The `urlpatterns` list routes URLs to views. For more information please see:
|
||||||
|
https://docs.djangoproject.com/en/1.8/topics/http/urls/
|
||||||
|
Examples:
|
||||||
|
Function views
|
||||||
|
1. Add an import: from my_app import views
|
||||||
|
2. Add a URL to urlpatterns: url(r'^$', views.home, name='home')
|
||||||
|
Class-based views
|
||||||
|
1. Add an import: from other_app.views import Home
|
||||||
|
2. Add a URL to urlpatterns: url(r'^$', Home.as_view(), name='home')
|
||||||
|
Including another URLconf
|
||||||
|
1. Add an import: from blog import urls as blog_urls
|
||||||
|
2. Add a URL to urlpatterns: url(r'^blog/', include(blog_urls))
|
||||||
|
"""
|
||||||
|
from django.conf.urls import include, url
|
||||||
|
from django.contrib import admin
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
url(r'^admin/', include(admin.site.urls)),
|
||||||
|
]
|
16
wordchallenge/wsgi.py
Normal file
16
wordchallenge/wsgi.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
"""
|
||||||
|
WSGI config for wordchallenge project.
|
||||||
|
|
||||||
|
It exposes the WSGI callable as a module-level variable named ``application``.
|
||||||
|
|
||||||
|
For more information on this file, see
|
||||||
|
https://docs.djangoproject.com/en/1.8/howto/deployment/wsgi/
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
|
||||||
|
from django.core.wsgi import get_wsgi_application
|
||||||
|
|
||||||
|
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "wordchallenge.settings")
|
||||||
|
|
||||||
|
application = get_wsgi_application()
|
0
words/__init__.py
Normal file
0
words/__init__.py
Normal file
37
words/migrations/0001_initial.py
Normal file
37
words/migrations/0001_initial.py
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.utils.timezone
|
||||||
|
from django.conf import settings
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Word',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True, serialize=False)),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='WordTranslation',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True, serialize=False)),
|
||||||
|
('language', models.CharField(db_index=True, max_length=5)),
|
||||||
|
('translation', models.CharField(null=True, max_length=100)),
|
||||||
|
('added_at', models.DateTimeField(default=django.utils.timezone.now)),
|
||||||
|
('added_by', models.ForeignKey(to=settings.AUTH_USER_MODEL)),
|
||||||
|
('word', models.ForeignKey(related_name='translations', to='words.Word')),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.AlterUniqueTogether(
|
||||||
|
name='wordtranslation',
|
||||||
|
unique_together=set([('language', 'translation'), ('word', 'language')]),
|
||||||
|
),
|
||||||
|
]
|
0
words/migrations/__init__.py
Normal file
0
words/migrations/__init__.py
Normal file
49
words/models.py
Normal file
49
words/models.py
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.auth.models import User
|
||||||
|
from django.db import models
|
||||||
|
from django.utils import timezone
|
||||||
|
from django.utils.translation import get_language
|
||||||
|
|
||||||
|
class Word(models.Model):
|
||||||
|
def translation(self, language):
|
||||||
|
try:
|
||||||
|
return self.translations.get(language=language)
|
||||||
|
except WordTranslation.DoesNotExist:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
try:
|
||||||
|
return self.translations.get(language=get_language()).translation
|
||||||
|
except WordTranslation.DoesNotExist:
|
||||||
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
return self.translations \
|
||||||
|
.get(language=settings.LANGUAGE_CODE).translation
|
||||||
|
except WordTranslation.DoesNotExist:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return ""
|
||||||
|
|
||||||
|
class WordTranslation(models.Model):
|
||||||
|
word = models.ForeignKey(Word, related_name='translations')
|
||||||
|
language = models.CharField(max_length=5, db_index=True)
|
||||||
|
translation = models.CharField(max_length=100, null=True, blank=False)
|
||||||
|
added_by = models.ForeignKey(User)
|
||||||
|
added_at = models.DateTimeField(default=timezone.now)
|
||||||
|
|
||||||
|
def clean(self):
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
|
||||||
|
if self.translation is None or self.translation == '':
|
||||||
|
raise ValidationError('translation must not be empty',
|
||||||
|
code='translation-empty')
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.translation
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = (
|
||||||
|
('word', 'language'),
|
||||||
|
('language', 'translation'),
|
||||||
|
)
|
56
words/tests.py
Normal file
56
words/tests.py
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
from django.contrib.auth.models import User
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
from django.utils.translation import activate
|
||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from .models import Word, WordTranslation
|
||||||
|
|
||||||
|
class WordTest(TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
user = User.objects.create_user(username='test', password='test')
|
||||||
|
self.word1 = Word.objects.create()
|
||||||
|
self.translation1 = WordTranslation.objects.create(
|
||||||
|
word=self.word1,
|
||||||
|
language='en-us',
|
||||||
|
translation='color',
|
||||||
|
added_by=user)
|
||||||
|
self.translation2 = WordTranslation.objects.create(
|
||||||
|
word=self.word1,
|
||||||
|
language='en-gb',
|
||||||
|
translation='colour',
|
||||||
|
added_by=user)
|
||||||
|
self.translation3 = WordTranslation.objects.create(
|
||||||
|
word=self.word1,
|
||||||
|
language='hu-hu',
|
||||||
|
translation='szín',
|
||||||
|
added_by=user)
|
||||||
|
|
||||||
|
def test_word_str(self):
|
||||||
|
with self.settings(LANGUAGE_CODE='en-us'):
|
||||||
|
self.assertEquals("color", self.word1.__str__())
|
||||||
|
|
||||||
|
with self.settings(LANGUAGE_CODE='en-gb'):
|
||||||
|
self.assertEquals('colour', self.word1.__str__())
|
||||||
|
|
||||||
|
activate('hu-hu')
|
||||||
|
self.assertEquals('szín', self.word1.__str__())
|
||||||
|
|
||||||
|
with self.settings(LANGUAGE_CODE='es-es'):
|
||||||
|
activate('is-is')
|
||||||
|
self.assertEquals('', self.word1.__str__())
|
||||||
|
|
||||||
|
def test_word_translation(self):
|
||||||
|
self.assertEquals('color', self.word1.translation('en-us').translation)
|
||||||
|
self.assertEquals('colour', self.word1.translation('en-gb').translation)
|
||||||
|
self.assertIsNone(self.word1.translation('is-is'))
|
||||||
|
|
||||||
|
def test_translation_validation(self):
|
||||||
|
word = WordTranslation()
|
||||||
|
|
||||||
|
with self.assertRaises(ValidationError) as ctx:
|
||||||
|
word.clean()
|
||||||
|
|
||||||
|
self.assertEquals('translation-empty', ctx.exception.code)
|
||||||
|
|
||||||
|
def test_translation_str(self):
|
||||||
|
self.assertEquals('color', self.translation1.__str__())
|
Loading…
Reference in New Issue
Block a user