107 lines
3.5 KiB
Python
107 lines
3.5 KiB
Python
from __future__ import absolute_import
|
|
|
|
import os
|
|
import tempfile
|
|
import re
|
|
import datetime
|
|
|
|
import slugify
|
|
|
|
try:
|
|
from cStringIO import StringIO
|
|
except ImportError:
|
|
from StringIO import StringIO
|
|
|
|
from django.db import models
|
|
from django.utils.translation import ugettext_lazy as _
|
|
from django.core import management
|
|
from django.core.files.uploadedfile import SimpleUploadedFile
|
|
|
|
from .literals import (FIXTURE_TYPES_CHOICES, FIXTURE_FILE_TYPE, COMMAND_LOADDATA,
|
|
BOOTSTRAP_EXTENSION, FIXTURE_METADATA_REMARK_CHARACTER)
|
|
from .managers import BootstrapSetupManager
|
|
from .classes import BootstrapModel, FixtureMetadata
|
|
|
|
|
|
class BootstrapSetup(models.Model):
|
|
"""
|
|
Model to store the fixture for a pre configured setup.
|
|
"""
|
|
name = models.CharField(max_length=128, verbose_name=_(u'name'), unique=True)
|
|
slug = models.SlugField(max_length=128, verbose_name=_(u'slug'), unique=True, blank=True)
|
|
description = models.TextField(verbose_name=_(u'description'), blank=True)
|
|
fixture = models.TextField(verbose_name=_(u'fixture'), help_text=_(u'These are the actual database structure creation instructions.'))
|
|
type = models.CharField(max_length=16, verbose_name=_(u'type'), choices=FIXTURE_TYPES_CHOICES)
|
|
created = models.DateTimeField(verbose_name=_('creation date and time'), default=lambda: datetime.datetime.now(), editable=False)
|
|
|
|
objects = BootstrapSetupManager()
|
|
|
|
def __unicode__(self):
|
|
return self.name
|
|
|
|
def get_extension(self):
|
|
"""
|
|
Return the fixture file extension based on the fixture type.
|
|
"""
|
|
return FIXTURE_FILE_TYPE[self.type]
|
|
|
|
def execute(self):
|
|
"""
|
|
Read a bootstrap's fixture and create the corresponding model
|
|
instances based on it.
|
|
"""
|
|
BootstrapModel.check_for_data()
|
|
handle, filepath = tempfile.mkstemp()
|
|
# Just need the filepath, close the file description
|
|
os.close(handle)
|
|
|
|
filepath = os.path.extsep.join([filepath, self.get_extension()])
|
|
|
|
with open(filepath, 'w') as file_handle:
|
|
file_handle.write(self.cleaned_fixture)
|
|
|
|
content = StringIO()
|
|
management.call_command(COMMAND_LOADDATA, filepath, verbosity=0, stderr=content)
|
|
content.seek(0, os.SEEK_END)
|
|
if content.tell():
|
|
content.seek(0)
|
|
raise Exception(content.readlines()[-2])
|
|
|
|
os.unlink(filepath)
|
|
|
|
@property
|
|
def cleaned_fixture(self):
|
|
"""
|
|
Return the bootstrap setup's fixture without comments.
|
|
"""
|
|
return re.sub(re.compile('%s.*?\n' % FIXTURE_METADATA_REMARK_CHARACTER), '', self.fixture)
|
|
|
|
def get_metadata_string(self):
|
|
"""
|
|
Return all the metadata for the current bootstrap fixture.
|
|
"""
|
|
return FixtureMetadata.generate_all(self)
|
|
|
|
def get_filename(self):
|
|
return os.extsep.join([self.name, BOOTSTRAP_EXTENSION])
|
|
|
|
def as_file(self):
|
|
return SimpleUploadedFile(name=self.get_filename(), content=self.fixture)
|
|
|
|
def save(self, *args, **kwargs):
|
|
update_metadata = kwargs.pop('update_metadata', True)
|
|
if update_metadata:
|
|
self.fixture = '%s\n%s\n%s' % (
|
|
BootstrapModel.get_magic_number(),
|
|
self.get_metadata_string(),
|
|
self.cleaned_fixture
|
|
)
|
|
if not self.slug:
|
|
self.slug = slugify.slugify(self.name)
|
|
return super(BootstrapSetup, self).save(*args, **kwargs)
|
|
|
|
class Meta:
|
|
verbose_name = _(u'bootstrap setup')
|
|
verbose_name_plural = _(u'bootstrap setups')
|
|
ordering = ['name']
|