diff --git a/lib/main/admin.py b/lib/main/admin.py index 0f074e8293..95d86e5f5c 100644 --- a/lib/main/admin.py +++ b/lib/main/admin.py @@ -16,6 +16,7 @@ import json +import urllib from django.conf.urls import * from django.contrib import admin @@ -51,7 +52,16 @@ admin.site.register(User, UserAdmin) # FIXME: Hide auth.Group admin -class OrganizationAdmin(admin.ModelAdmin): +class BaseModelAdmin(admin.ModelAdmin): + + def save_model(self, request, obj, form, change): + # Automatically set created_by when saved from the admin. + # FIXME: Doesn't handle inline model instances yet. + if hasattr(obj, 'created_by') and obj.created_by is None: + obj.created_by = request.user + return super(BaseModelAdmin, self).save_model(request, obj, form, change) + +class OrganizationAdmin(BaseModelAdmin): list_display = ('name', 'description', 'active') list_filter = ('active', 'tags') @@ -62,7 +72,7 @@ class OrganizationAdmin(admin.ModelAdmin): (_('Tags'), {'fields': ('tags',)}), (_('Audit Trail'), {'fields': ('creation_date', 'audit_trail',)}), ) - readonly_fields = ('creation_date', 'audit_trail') + readonly_fields = ('creation_date', 'created_by', 'audit_trail') filter_horizontal = ('users', 'admins', 'projects', 'tags') class InventoryHostInline(admin.StackedInline): @@ -79,7 +89,7 @@ class InventoryGroupInline(admin.StackedInline): fields = ('name', 'description', 'active', 'parents', 'hosts', 'tags') filter_horizontal = ('parents', 'hosts', 'tags') -class InventoryAdmin(admin.ModelAdmin): +class InventoryAdmin(BaseModelAdmin): list_display = ('name', 'organization', 'description', 'active') list_filter = ('organization', 'active') @@ -89,11 +99,11 @@ class InventoryAdmin(admin.ModelAdmin): (_('Tags'), {'fields': ('tags',)}), (_('Audit Trail'), {'fields': ('creation_date', 'audit_trail',)}), ) - readonly_fields = ('creation_date', 'audit_trail') + readonly_fields = ('creation_date', 'created_by', 'audit_trail') filter_horizontal = ('tags',) inlines = [InventoryHostInline, InventoryGroupInline] -class TagAdmin(admin.ModelAdmin): +class TagAdmin(BaseModelAdmin): list_display = ('name',) @@ -111,18 +121,18 @@ class VariableDataInline(admin.StackedInline): # FIXME: Doesn't yet work as inline due to the way the OneToOne field is # defined. -class LaunchJobHostSummaryInline(admin.TabularInline): +class JobHostSummaryInline(admin.TabularInline): - model = LaunchJobHostSummary + model = JobHostSummary extra = 0 can_delete = False def has_add_permission(self, request): return False -class LaunchJobStatusEventInline(admin.StackedInline): +class JobEventInline(admin.StackedInline): - model = LaunchJobStatusEvent + model = JobEvent extra = 0 can_delete = False @@ -130,161 +140,197 @@ class LaunchJobStatusEventInline(admin.StackedInline): return False def get_event_data_display(self, obj): - return format_html('
{0}', json.dumps(obj.event_data, indent=4))
+ return format_html('{0}',
+ json.dumps(obj.event_data, indent=4))
get_event_data_display.short_description = _('Event data')
get_event_data_display.allow_tags = True
-class LaunchJobHostSummaryInlineForHost(LaunchJobHostSummaryInline):
+class JobHostSummaryInlineForHost(JobHostSummaryInline):
- fields = ('launch_job_status', 'changed', 'dark', 'failures', 'ok', 'processed', 'skipped')
- readonly_fields = ('launch_job_status', 'changed', 'dark', 'failures', 'ok', 'processed', 'skipped')
+ fields = ('job', 'changed', 'dark', 'failures', 'ok', 'processed',
+ 'skipped')
+ readonly_fields = ('job', 'changed', 'dark', 'failures', 'ok', 'processed',
+ 'skipped')
-class LaunchJobStatusEventInlineForHost(LaunchJobStatusEventInline):
+class JobEventInlineForHost(JobEventInline):
- fields = ('created', 'event', 'get_event_data_display', 'launch_job_status')
- readonly_fields = ('created', 'event', 'get_event_data_display', 'launch_job_status')
+ fields = ('job', 'created', 'event', 'get_event_data_display')
+ readonly_fields = ('job', 'created', 'event', 'get_event_data_display')
-class HostAdmin(admin.ModelAdmin):
+class HostAdmin(BaseModelAdmin):
list_display = ('name', 'inventory', 'description', 'active')
list_filter = ('inventory', 'active')
fields = ('name', 'inventory', 'description', 'active', 'tags',
'created_by', 'audit_trail')
+ readonly_fields = ('creation_date', 'created_by', 'audit_trail')
filter_horizontal = ('tags',)
# FIXME: Edit reverse of many to many for groups.
#inlines = [VariableDataInline]
- inlines = [LaunchJobHostSummaryInlineForHost, LaunchJobStatusEventInlineForHost]
+ inlines = [JobHostSummaryInlineForHost, JobEventInlineForHost]
-class GroupAdmin(admin.ModelAdmin):
+class GroupAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('parents', 'hosts', 'tags')
#inlines = [VariableDataInline]
-class VariableDataAdmin(admin.ModelAdmin):
+class VariableDataAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('tags',)
-class CredentialAdmin(admin.ModelAdmin):
+class CredentialAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('tags',)
-class TeamAdmin(admin.ModelAdmin):
+class TeamAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('projects', 'users', 'tags')
-class ProjectAdmin(admin.ModelAdmin):
+class ProjectAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('tags',)
-class PermissionAdmin(admin.ModelAdmin):
+class PermissionAdmin(BaseModelAdmin):
list_display = ('name', 'description', 'active')
filter_horizontal = ('tags',)
-class LaunchJobAdmin(admin.ModelAdmin):
+class JobTemplateAdmin(BaseModelAdmin):
- list_display = ('name', 'description', 'active', 'get_start_link_display',
- 'get_statuses_link_display')
+ list_display = ('name', 'description', 'active', 'get_create_link_display',
+ 'get_jobs_link_display')
fieldsets = (
- (None, {'fields': ('name', 'active', 'created_by', 'description',
- 'get_start_link_display', 'get_statuses_link_display')}),
+ (None, {'fields': ('name', 'active', 'description',
+ 'get_create_link_display', 'get_jobs_link_display')}),
(_('Job Parameters'), {'fields': ('inventory', 'project', 'credential',
'user', 'job_type')}),
- (_('Tags'), {'fields': ('tags',)}),
- (_('Audit Trail'), {'fields': ('creation_date', 'audit_trail',)}),
+ #(_('Tags'), {'fields': ('tags',)}),
+ (_('Audit Trail'), {'fields': ('creation_date', 'created_by',
+ 'audit_trail',)}),
)
- readonly_fields = ('creation_date', 'audit_trail', 'get_start_link_display',
- 'get_statuses_link_display')
- filter_horizontal = ('tags',)
+ readonly_fields = ('creation_date', 'created_by', 'audit_trail',
+ 'get_create_link_display', 'get_jobs_link_display')
+ #filter_horizontal = ('tags',)
- def get_start_link_display(self, obj):
- info = self.model._meta.app_label, self.model._meta.module_name
- start_url = reverse('admin:%s_%s_start' % info, args=(obj.pk,),
- current_app=self.admin_site.name)
- return 'Run Job' % start_url
- get_start_link_display.short_description = _('Run')
- get_start_link_display.allow_tags = True
-
- def get_statuses_link_display(self, obj):
- info = LaunchJobStatus._meta.app_label, LaunchJobStatus._meta.module_name
- statuses_url = reverse('admin:%s_%s_changelist' % info,
- current_app=self.admin_site.name)
- statuses_url += '?launch_job__id__exact=%d' % obj.pk
- return 'View Logs' % statuses_url
- get_statuses_link_display.short_description = _('Logs')
- get_statuses_link_display.allow_tags = True
-
- def get_urls(self):
- info = self.model._meta.app_label, self.model._meta.module_name
- urls = super(LaunchJobAdmin, self).get_urls()
- return patterns('',
- url(r'^(.+)/start/$',
- self.admin_site.admin_view(self.start_job_view),
- name='%s_%s_start' % info),
- ) + urls
-
- def start_job_view(self, request, object_id):
- obj = self.get_object(request, unquote(object_id))
- ljs = obj.start()
- info = ljs._meta.app_label, ljs._meta.module_name
- status_url = reverse('admin:%s_%s_change' % info, args=(ljs.pk,),
+ def get_create_link_display(self, obj):
+ info = Job._meta.app_label, Job._meta.module_name
+ create_url = reverse('admin:%s_%s_add' % info,
current_app=self.admin_site.name)
- messages.success(request, '%s has been started.' % ljs)
- return HttpResponseRedirect(status_url)
+ create_opts = {
+ 'job_template': obj.pk,
+ 'job_type': obj.job_type,
+ }
+ if obj.inventory:
+ create_opts['inventory'] = obj.inventory.pk
+ if obj.project:
+ create_opts['project'] = obj.project.pk
+ if obj.credential:
+ create_opts['credential'] = obj.credential.pk
+ if obj.user:
+ create_opts['user'] = obj.user.pk
+ create_url += '?%s' % urllib.urlencode(create_opts)
+ return format_html('{1}', create_url, 'Create Job')
+ get_create_link_display.short_description = _('Create Job')
+ get_create_link_display.allow_tags = True
-class LaunchJobHostSummaryInlineForLaunchJobStatus(LaunchJobHostSummaryInline):
+ def get_jobs_link_display(self, obj):
+ info = Job._meta.app_label, Job._meta.module_name
+ jobs_url = reverse('admin:%s_%s_changelist' % info,
+ current_app=self.admin_site.name)
+ jobs_url += '?job_template__id__exact=%d' % obj.pk
+ return format_html('{1}', jobs_url, 'View Jobs')
+ get_jobs_link_display.short_description = _('View Jobs')
+ get_jobs_link_display.allow_tags = True
- fields = ('host', 'changed', 'dark', 'failures', 'ok', 'processed', 'skipped')
- readonly_fields = ('host', 'changed', 'dark', 'failures', 'ok', 'processed', 'skipped')
+class JobHostSummaryInlineForJob(JobHostSummaryInline):
-class LaunchJobStatusEventInlineForLaunchJobStatus(LaunchJobStatusEventInline):
+ fields = ('host', 'changed', 'dark', 'failures', 'ok', 'processed',
+ 'skipped')
+ readonly_fields = ('host', 'changed', 'dark', 'failures', 'ok',
+ 'processed', 'skipped')
+
+class JobEventInlineForJob(JobEventInline):
fields = ('created', 'event', 'get_event_data_display', 'host')
readonly_fields = ('created', 'event', 'get_event_data_display', 'host')
-class LaunchJobStatusAdmin(admin.ModelAdmin):
+class JobAdmin(BaseModelAdmin):
- list_display = ('name', 'launch_job', 'status')
- fields = ('name', 'get_launch_job_display', 'status',
- 'get_result_stdout_display', 'get_result_stderr_display',
- 'get_result_traceback_display', 'celery_task_id', 'tags',
- 'created_by')
- readonly_fields = ('name', 'description', 'status', 'get_launch_job_display',
+ list_display = ('name', 'job_template', 'status')
+ fieldsets = (
+ (None, {'fields': ('name', 'job_template', 'description')}),
+ (_('Job Parameters'), {'fields': ('inventory', 'project', 'credential',
+ 'user', 'job_type')}),
+ #(_('Tags'), {'fields': ('tags',)}),
+ (_('Audit Trail'), {'fields': ('creation_date', 'created_by',
+ 'audit_trail',)}),
+ (_('Job Status'), {'fields': ('status', 'get_result_stdout_display',
+ 'get_result_stderr_display',
+ 'get_result_traceback_display',
+ 'celery_task_id')}),
+ )
+ readonly_fields = ('status', 'get_job_template_display',
'get_result_stdout_display', 'get_result_stderr_display',
'get_result_traceback_display', 'celery_task_id',
- 'created_by', 'tags', 'audit_trail', 'active')
+ 'creation_date', 'created_by', 'audit_trail',)
filter_horizontal = ('tags',)
- inlines = [LaunchJobHostSummaryInlineForLaunchJobStatus,
- LaunchJobStatusEventInlineForLaunchJobStatus]
+ inlines = [JobHostSummaryInlineForJob, JobEventInlineForJob]
- def has_add_permission(self, request):
- return False
+ def get_readonly_fields(self, request, obj=None):
+ ro_fields = list(super(JobAdmin, self).get_readonly_fields(request, obj))
+ if obj and obj.pk:
+ ro_fields.extend(['name', 'description', 'job_template',
+ 'inventory', 'project', 'credential', 'user',
+ 'job_type'])
+ return ro_fields
- def get_launch_job_display(self, obj):
- info = obj.launch_job._meta.app_label, obj.launch_job._meta.module_name
- lj_url = reverse('admin:%s_%s_change' % info, args=(obj.launch_job.pk,),
- current_app=self.admin_site.name)
- return format_html('{1}', lj_url, obj.launch_job)
- get_launch_job_display.short_description = _('Launch job')
- get_launch_job_display.allow_tags = True
+ def get_fieldsets(self, request, obj=None):
+ fsets = list(super(JobAdmin, self).get_fieldsets(request, obj))
+ if not obj or not obj.pk:
+ fsets = [fs for fs in fsets if
+ 'creation_date' not in fs[1]['fields'] and
+ 'status' not in fs[1]['fields']]
+ return fsets
+
+ def get_inline_instances(self, request, obj=None):
+ if obj and obj.pk:
+ return super(JobAdmin, self).get_inline_instances(request, obj)
+ else:
+ return []
+
+ def get_job_template_display(self, obj):
+ if obj.job_template:
+ info = JobTemplate._meta.app_label, JobTemplate._meta.module_name
+ job_template_url = reverse('admin:%s_%s_change' % info,
+ args=(obj.job_template.pk,),
+ current_app=self.admin_site.name)
+ return format_html('{1}', job_template_url,
+ obj.job_template)
+ else:
+ return _('(None)')
+ get_job_template_display.short_description = _('Job template')
+ get_job_template_display.allow_tags = True
def get_result_stdout_display(self, obj):
- return format_html('{0}', obj.result_stdout or ' ')
+ return format_html('{0}',
+ obj.result_stdout or ' ')
get_result_stdout_display.short_description = _('Stdout')
get_result_stdout_display.allow_tags = True
def get_result_stderr_display(self, obj):
- return format_html('{0}', obj.result_stderr or ' ')
+ return format_html('{0}',
+ obj.result_stderr or ' ')
get_result_stderr_display.short_description = _('Stderr')
get_result_stderr_display.allow_tags = True
def get_result_traceback_display(self, obj):
- return format_html('{0}', obj.result_traceback or ' ')
+ return format_html('{0}',
+ obj.result_traceback or ' ')
get_result_traceback_display.short_description = _('Traceback')
get_result_traceback_display.allow_tags = True
@@ -300,5 +346,5 @@ admin.site.register(VariableData, VariableDataAdmin)
admin.site.register(Team, TeamAdmin)
admin.site.register(Project, ProjectAdmin)
admin.site.register(Credential, CredentialAdmin)
-admin.site.register(LaunchJob, LaunchJobAdmin)
-admin.site.register(LaunchJobStatus, LaunchJobStatusAdmin)
+admin.site.register(JobTemplate, JobTemplateAdmin)
+admin.site.register(Job, JobAdmin)
diff --git a/lib/main/management/commands/acom_callback_event.py b/lib/main/management/commands/acom_callback_event.py
index 5485ff4417..040b9475a7 100755
--- a/lib/main/management/commands/acom_callback_event.py
+++ b/lib/main/management/commands/acom_callback_event.py
@@ -30,10 +30,10 @@ class Command(NoArgsCommand):
help = 'Ansible Commander Callback Event Capture'
option_list = NoArgsCommand.option_list + (
- make_option('-i', '--launch-job-status', dest='launch_job_status_id',
+ make_option('-j', '--job', dest='job_id',
type='int', default=0,
- help='Launch job status ID (can also be specified using '
- 'ACOM_LAUNCH_JOB_STATUS_ID environment variable)'),
+ help='Job ID (can also be specified using ACOM_JOB_ID '
+ 'environment variable)'),
make_option('-e', '--event', dest='event_type', default=None,
help='Event type'),
make_option('-f', '--file', dest='event_data_file', default=None,
@@ -44,29 +44,28 @@ class Command(NoArgsCommand):
)
def handle_noargs(self, **options):
- from lib.main.models import LaunchJobStatus, LaunchJobStatusEvent
+ from lib.main.models import Job, JobEvent
event_type = options.get('event_type', None)
if not event_type:
raise CommandError('No event specified')
- if event_type not in [x[0] for x in LaunchJobStatusEvent.EVENT_TYPES]:
+ if event_type not in [x[0] for x in JobEvent.EVENT_TYPES]:
raise CommandError('Unsupported event')
event_data_file = options.get('event_data_file', None)
event_data_json = options.get('event_data_json', None)
if event_data_file is None and event_data_json is None:
raise CommandError('Either --file or --data must be specified')
try:
- launch_job_status_id = int(os.getenv('ACOM_LAUNCH_JOB_STATUS_ID',
- options.get('launch_job_status_id', 0)))
+ job_id = int(os.getenv('ACOM_JOB_ID', options.get('job_id', 0)))
except ValueError:
- raise CommandError('Launch job status ID must be an integer')
- if not launch_job_status_id:
- raise CommandError('No launch job status ID specified')
+ raise CommandError('Job ID must be an integer')
+ if not job_id:
+ raise CommandError('No Job ID specified')
try:
- launch_job_status = LaunchJobStatus.objects.get(id=launch_job_status_id)
- except LaunchJobStatus.DoesNotExist:
- raise CommandError('Launch job status with ID %d not found' % launch_job_status_id)
- if launch_job_status.status != 'running':
- raise CommandError('Unable to add event except when launch job is running')
+ job = Job.objects.get(id=job_id)
+ except Job.DoesNotExist:
+ raise CommandError('Job with ID %d not found' % job_id)
+ if job.status != 'running':
+ raise CommandError('Unable to add event except when job is running')
try:
if event_data_json is None:
try:
@@ -81,8 +80,7 @@ class Command(NoArgsCommand):
event_data = json.loads(event_data_json)
except ValueError:
raise CommandError('Error parsing JSON data')
- launch_job_status.launch_job_status_events.create(event=event_type,
- event_data=event_data)
+ job.job_events.create(event=event_type, event_data=event_data)
if __name__ == '__main__':
from __init__ import run_command_as_script
diff --git a/lib/main/management/commands/acom_inventory.py b/lib/main/management/commands/acom_inventory.py
index 4f5f06840f..bd723722f6 100755
--- a/lib/main/management/commands/acom_inventory.py
+++ b/lib/main/management/commands/acom_inventory.py
@@ -27,9 +27,9 @@ class Command(NoArgsCommand):
help = 'Ansible Commander Inventory script'
option_list = NoArgsCommand.option_list + (
- make_option('-i', '--inventory', dest='inventory', type='int', default=0,
- help='Inventory ID (can also be specified using '
- 'ACOM_INVENTORY_ID environment variable)'),
+ make_option('-i', '--inventory', dest='inventory_id', type='int',
+ default=0, help='Inventory ID (can also be specified using'
+ ' ACOM_INVENTORY_ID environment variable)'),
make_option('--list', action='store_true', dest='list', default=False,
help='Return JSON hash of host groups.'),
make_option('--host', dest='host', default='',
@@ -75,7 +75,7 @@ class Command(NoArgsCommand):
try:
# Command line argument takes precedence over environment
# variable.
- inventory_id = int(options.get('inventory', 0) or \
+ inventory_id = int(options.get('inventory_id', 0) or \
os.getenv('ACOM_INVENTORY_ID', 0))
except ValueError:
raise CommandError('Inventory ID must be an integer')
diff --git a/lib/main/migrations/0013_changes.py b/lib/main/migrations/0013_changes.py
new file mode 100644
index 0000000000..8b750e348a
--- /dev/null
+++ b/lib/main/migrations/0013_changes.py
@@ -0,0 +1,501 @@
+# -*- coding: utf-8 -*-
+import datetime
+from south.db import db
+from south.v2 import SchemaMigration
+from django.db import models
+
+
+class Migration(SchemaMigration):
+
+ def forwards(self, orm):
+ # Removing unique constraint on 'LaunchJobHostSummary', fields ['launch_job_status', 'host']
+ db.delete_unique(u'main_launchjobhostsummary', ['launch_job_status_id', 'host_id'])
+
+ # Deleting model 'LaunchJobHostSummary'
+ db.delete_table(u'main_launchjobhostsummary')
+
+ # Deleting model 'LaunchJobStatusEvent'
+ db.delete_table(u'main_launchjobstatusevent')
+
+ # Deleting model 'LaunchJob'
+ db.delete_table(u'main_launchjob')
+
+ # Removing M2M table for field tags on 'LaunchJob'
+ db.delete_table('main_launchjob_tags')
+
+ # Removing M2M table for field audit_trail on 'LaunchJob'
+ db.delete_table('main_launchjob_audit_trail')
+
+ # Deleting model 'LaunchJobStatus'
+ db.delete_table(u'main_launchjobstatus')
+
+ # Removing M2M table for field tags on 'LaunchJobStatus'
+ db.delete_table('main_launchjobstatus_tags')
+
+ # Removing M2M table for field audit_trail on 'LaunchJobStatus'
+ db.delete_table('main_launchjobstatus_audit_trail')
+
+ # Adding model 'Job'
+ db.create_table(u'main_job', (
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('description', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('created_by', self.gf('django.db.models.fields.related.ForeignKey')(related_name="{'class': 'job', 'app_label': 'main'}(class)s_created", null=True, on_delete=models.SET_NULL, to=orm['auth.User'])),
+ ('creation_date', self.gf('django.db.models.fields.DateField')(auto_now_add=True, blank=True)),
+ ('active', self.gf('django.db.models.fields.BooleanField')(default=True)),
+ ('name', self.gf('django.db.models.fields.CharField')(unique=True, max_length=512)),
+ ('job_template', self.gf('django.db.models.fields.related.ForeignKey')(related_name='jobs', on_delete=models.SET_NULL, default=None, to=orm['main.JobTemplate'], blank=True, null=True)),
+ ('job_type', self.gf('django.db.models.fields.CharField')(max_length=64)),
+ ('inventory', self.gf('django.db.models.fields.related.ForeignKey')(related_name='jobs', null=True, on_delete=models.SET_NULL, to=orm['main.Inventory'])),
+ ('credential', self.gf('django.db.models.fields.related.ForeignKey')(related_name='jobs', null=True, on_delete=models.SET_NULL, to=orm['main.Credential'])),
+ ('project', self.gf('django.db.models.fields.related.ForeignKey')(related_name='jobs', null=True, on_delete=models.SET_NULL, to=orm['main.Project'])),
+ ('user', self.gf('django.db.models.fields.related.ForeignKey')(related_name='jobs', null=True, on_delete=models.SET_NULL, to=orm['auth.User'])),
+ ('status', self.gf('django.db.models.fields.CharField')(default='pending', max_length=20)),
+ ('result_stdout', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('result_stderr', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('result_traceback', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('celery_task_id', self.gf('django.db.models.fields.CharField')(default='', max_length=100, blank=True)),
+ ))
+ db.send_create_signal('main', ['Job'])
+
+ # Adding M2M table for field tags on 'Job'
+ db.create_table(u'main_job_tags', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('job', models.ForeignKey(orm['main.job'], null=False)),
+ ('tag', models.ForeignKey(orm['main.tag'], null=False))
+ ))
+ db.create_unique(u'main_job_tags', ['job_id', 'tag_id'])
+
+ # Adding M2M table for field audit_trail on 'Job'
+ db.create_table(u'main_job_audit_trail', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('job', models.ForeignKey(orm['main.job'], null=False)),
+ ('audittrail', models.ForeignKey(orm['main.audittrail'], null=False))
+ ))
+ db.create_unique(u'main_job_audit_trail', ['job_id', 'audittrail_id'])
+
+ # Adding model 'JobHostSummary'
+ db.create_table(u'main_jobhostsummary', (
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('job', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_host_summaries', to=orm['main.Job'])),
+ ('host', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_host_summaries', to=orm['main.Host'])),
+ ('changed', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('dark', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('failures', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('ok', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('processed', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('skipped', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ))
+ db.send_create_signal(u'main', ['JobHostSummary'])
+
+ # Adding unique constraint on 'JobHostSummary', fields ['job', 'host']
+ db.create_unique(u'main_jobhostsummary', ['job_id', 'host_id'])
+
+ # Adding model 'JobTemplate'
+ db.create_table(u'main_jobtemplate', (
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('description', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('created_by', self.gf('django.db.models.fields.related.ForeignKey')(related_name="{'class': 'jobtemplate', 'app_label': 'main'}(class)s_created", null=True, on_delete=models.SET_NULL, to=orm['auth.User'])),
+ ('creation_date', self.gf('django.db.models.fields.DateField')(auto_now_add=True, blank=True)),
+ ('active', self.gf('django.db.models.fields.BooleanField')(default=True)),
+ ('name', self.gf('django.db.models.fields.CharField')(unique=True, max_length=512)),
+ ('job_type', self.gf('django.db.models.fields.CharField')(max_length=64)),
+ ('inventory', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_templates', on_delete=models.SET_NULL, default=None, to=orm['main.Inventory'], blank=True, null=True)),
+ ('credential', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_templates', on_delete=models.SET_NULL, default=None, to=orm['main.Credential'], blank=True, null=True)),
+ ('project', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_templates', on_delete=models.SET_NULL, default=None, to=orm['main.Project'], blank=True, null=True)),
+ ('user', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_templates', on_delete=models.SET_NULL, default=None, to=orm['auth.User'], blank=True, null=True)),
+ ))
+ db.send_create_signal('main', ['JobTemplate'])
+
+ # Adding M2M table for field tags on 'JobTemplate'
+ db.create_table(u'main_jobtemplate_tags', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('jobtemplate', models.ForeignKey(orm['main.jobtemplate'], null=False)),
+ ('tag', models.ForeignKey(orm['main.tag'], null=False))
+ ))
+ db.create_unique(u'main_jobtemplate_tags', ['jobtemplate_id', 'tag_id'])
+
+ # Adding M2M table for field audit_trail on 'JobTemplate'
+ db.create_table(u'main_jobtemplate_audit_trail', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('jobtemplate', models.ForeignKey(orm['main.jobtemplate'], null=False)),
+ ('audittrail', models.ForeignKey(orm['main.audittrail'], null=False))
+ ))
+ db.create_unique(u'main_jobtemplate_audit_trail', ['jobtemplate_id', 'audittrail_id'])
+
+ # Adding model 'JobEvent'
+ db.create_table(u'main_jobevent', (
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('job', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_events', to=orm['main.Job'])),
+ ('created', self.gf('django.db.models.fields.DateTimeField')(auto_now_add=True, blank=True)),
+ ('event', self.gf('django.db.models.fields.CharField')(max_length=100)),
+ ('event_data', self.gf('jsonfield.fields.JSONField')(default='', blank=True)),
+ ('host', self.gf('django.db.models.fields.related.ForeignKey')(related_name='job_events', on_delete=models.SET_NULL, default=None, to=orm['main.Host'], blank=True, null=True)),
+ ))
+ db.send_create_signal('main', ['JobEvent'])
+
+
+ def backwards(self, orm):
+ # Removing unique constraint on 'JobHostSummary', fields ['job', 'host']
+ db.delete_unique(u'main_jobhostsummary', ['job_id', 'host_id'])
+
+ # Adding model 'LaunchJobHostSummary'
+ db.create_table(u'main_launchjobhostsummary', (
+ ('dark', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('skipped', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('host', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_host_summaries', to=orm['main.Host'])),
+ ('ok', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('processed', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('failures', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ ('changed', self.gf('django.db.models.fields.PositiveIntegerField')(default=0)),
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('launch_job_status', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_host_summaries', to=orm['main.LaunchJobStatus'])),
+ ))
+ db.send_create_signal(u'main', ['LaunchJobHostSummary'])
+
+ # Adding unique constraint on 'LaunchJobHostSummary', fields ['launch_job_status', 'host']
+ db.create_unique(u'main_launchjobhostsummary', ['launch_job_status_id', 'host_id'])
+
+ # Adding model 'LaunchJobStatusEvent'
+ db.create_table(u'main_launchjobstatusevent', (
+ ('event', self.gf('django.db.models.fields.CharField')(max_length=100)),
+ ('created', self.gf('django.db.models.fields.DateTimeField')(auto_now_add=True, blank=True)),
+ ('event_data', self.gf('jsonfield.fields.JSONField')(default='', blank=True)),
+ ('host', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_status_events', on_delete=models.SET_NULL, default=None, to=orm['main.Host'], blank=True, null=True)),
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('launch_job_status', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_status_events', to=orm['main.LaunchJobStatus'])),
+ ))
+ db.send_create_signal('main', ['LaunchJobStatusEvent'])
+
+ # Adding model 'LaunchJob'
+ db.create_table(u'main_launchjob', (
+ ('credential', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_jobs', on_delete=models.SET_NULL, default=None, to=orm['main.Credential'], blank=True, null=True)),
+ ('description', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('job_type', self.gf('django.db.models.fields.CharField')(max_length=64)),
+ ('creation_date', self.gf('django.db.models.fields.DateField')(auto_now_add=True, blank=True)),
+ ('user', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_jobs', on_delete=models.SET_NULL, default=None, to=orm['auth.User'], blank=True, null=True)),
+ ('active', self.gf('django.db.models.fields.BooleanField')(default=True)),
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('name', self.gf('django.db.models.fields.CharField')(max_length=512, unique=True)),
+ ('created_by', self.gf('django.db.models.fields.related.ForeignKey')(related_name="{'class': 'launchjob', 'app_label': 'main'}(class)s_created", null=True, on_delete=models.SET_NULL, to=orm['auth.User'])),
+ ('project', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_jobs', on_delete=models.SET_NULL, default=None, to=orm['main.Project'], blank=True, null=True)),
+ ('inventory', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_jobs', on_delete=models.SET_NULL, default=None, to=orm['main.Inventory'], blank=True, null=True)),
+ ))
+ db.send_create_signal('main', ['LaunchJob'])
+
+ # Adding M2M table for field tags on 'LaunchJob'
+ db.create_table(u'main_launchjob_tags', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('launchjob', models.ForeignKey(orm['main.launchjob'], null=False)),
+ ('tag', models.ForeignKey(orm['main.tag'], null=False))
+ ))
+ db.create_unique(u'main_launchjob_tags', ['launchjob_id', 'tag_id'])
+
+ # Adding M2M table for field audit_trail on 'LaunchJob'
+ db.create_table(u'main_launchjob_audit_trail', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('launchjob', models.ForeignKey(orm['main.launchjob'], null=False)),
+ ('audittrail', models.ForeignKey(orm['main.audittrail'], null=False))
+ ))
+ db.create_unique(u'main_launchjob_audit_trail', ['launchjob_id', 'audittrail_id'])
+
+ # Adding model 'LaunchJobStatus'
+ db.create_table(u'main_launchjobstatus', (
+ ('status', self.gf('django.db.models.fields.CharField')(default='pending', max_length=20)),
+ ('description', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('result_traceback', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('result_stdout', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('created_by', self.gf('django.db.models.fields.related.ForeignKey')(related_name="{'class': 'launchjobstatus', 'app_label': 'main'}(class)s_created", null=True, on_delete=models.SET_NULL, to=orm['auth.User'])),
+ ('result_stderr', self.gf('django.db.models.fields.TextField')(default='', blank=True)),
+ ('celery_task_id', self.gf('django.db.models.fields.CharField')(default='', max_length=100, blank=True)),
+ ('launch_job', self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_statuses', null=True, on_delete=models.SET_NULL, to=orm['main.LaunchJob'])),
+ ('active', self.gf('django.db.models.fields.BooleanField')(default=True)),
+ ('creation_date', self.gf('django.db.models.fields.DateField')(auto_now_add=True, blank=True)),
+ (u'id', self.gf('django.db.models.fields.AutoField')(primary_key=True)),
+ ('name', self.gf('django.db.models.fields.CharField')(max_length=512, unique=True)),
+ ))
+ db.send_create_signal('main', ['LaunchJobStatus'])
+
+ # Adding M2M table for field tags on 'LaunchJobStatus'
+ db.create_table(u'main_launchjobstatus_tags', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('launchjobstatus', models.ForeignKey(orm['main.launchjobstatus'], null=False)),
+ ('tag', models.ForeignKey(orm['main.tag'], null=False))
+ ))
+ db.create_unique(u'main_launchjobstatus_tags', ['launchjobstatus_id', 'tag_id'])
+
+ # Adding M2M table for field audit_trail on 'LaunchJobStatus'
+ db.create_table(u'main_launchjobstatus_audit_trail', (
+ ('id', models.AutoField(verbose_name='ID', primary_key=True, auto_created=True)),
+ ('launchjobstatus', models.ForeignKey(orm['main.launchjobstatus'], null=False)),
+ ('audittrail', models.ForeignKey(orm['main.audittrail'], null=False))
+ ))
+ db.create_unique(u'main_launchjobstatus_audit_trail', ['launchjobstatus_id', 'audittrail_id'])
+
+ # Deleting model 'Job'
+ db.delete_table(u'main_job')
+
+ # Removing M2M table for field tags on 'Job'
+ db.delete_table('main_job_tags')
+
+ # Removing M2M table for field audit_trail on 'Job'
+ db.delete_table('main_job_audit_trail')
+
+ # Deleting model 'JobHostSummary'
+ db.delete_table(u'main_jobhostsummary')
+
+ # Deleting model 'JobTemplate'
+ db.delete_table(u'main_jobtemplate')
+
+ # Removing M2M table for field tags on 'JobTemplate'
+ db.delete_table('main_jobtemplate_tags')
+
+ # Removing M2M table for field audit_trail on 'JobTemplate'
+ db.delete_table('main_jobtemplate_audit_trail')
+
+ # Deleting model 'JobEvent'
+ db.delete_table(u'main_jobevent')
+
+
+ models = {
+ u'auth.group': {
+ 'Meta': {'object_name': 'Group'},
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '80'}),
+ 'permissions': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['auth.Permission']", 'symmetrical': 'False', 'blank': 'True'})
+ },
+ u'auth.permission': {
+ 'Meta': {'ordering': "(u'content_type__app_label', u'content_type__model', u'codename')", 'unique_together': "((u'content_type', u'codename'),)", 'object_name': 'Permission'},
+ 'codename': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
+ 'content_type': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['contenttypes.ContentType']"}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '50'})
+ },
+ u'auth.user': {
+ 'Meta': {'object_name': 'User'},
+ 'date_joined': ('django.db.models.fields.DateTimeField', [], {'default': 'datetime.datetime.now'}),
+ 'email': ('django.db.models.fields.EmailField', [], {'max_length': '75', 'blank': 'True'}),
+ 'first_name': ('django.db.models.fields.CharField', [], {'max_length': '30', 'blank': 'True'}),
+ 'groups': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['auth.Group']", 'symmetrical': 'False', 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'is_active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'is_staff': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
+ 'is_superuser': ('django.db.models.fields.BooleanField', [], {'default': 'False'}),
+ 'last_login': ('django.db.models.fields.DateTimeField', [], {'default': 'datetime.datetime.now'}),
+ 'last_name': ('django.db.models.fields.CharField', [], {'max_length': '30', 'blank': 'True'}),
+ 'password': ('django.db.models.fields.CharField', [], {'max_length': '128'}),
+ 'user_permissions': ('django.db.models.fields.related.ManyToManyField', [], {'to': u"orm['auth.Permission']", 'symmetrical': 'False', 'blank': 'True'}),
+ 'username': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '30'})
+ },
+ u'contenttypes.contenttype': {
+ 'Meta': {'ordering': "('name',)", 'unique_together': "(('app_label', 'model'),)", 'object_name': 'ContentType', 'db_table': "'django_content_type'"},
+ 'app_label': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'model': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '100'})
+ },
+ 'main.audittrail': {
+ 'Meta': {'object_name': 'AuditTrail'},
+ 'comment': ('django.db.models.fields.TextField', [], {}),
+ 'delta': ('django.db.models.fields.TextField', [], {}),
+ 'detail': ('django.db.models.fields.TextField', [], {}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'modified_by': ('django.db.models.fields.related.ForeignKey', [], {'to': u"orm['auth.User']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'}),
+ 'resource_type': ('django.db.models.fields.CharField', [], {'max_length': '64'}),
+ 'tag': ('django.db.models.fields.related.ForeignKey', [], {'to': "orm['main.Tag']", 'null': 'True', 'on_delete': 'models.SET_NULL', 'blank': 'True'})
+ },
+ 'main.credential': {
+ 'Meta': {'object_name': 'Credential'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'credential_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'credential\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'default_username': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '1024', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'}),
+ 'ssh_key_data': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'ssh_key_unlock': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '1024', 'blank': 'True'}),
+ 'ssh_password': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '1024', 'blank': 'True'}),
+ 'sudo_password': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '1024', 'blank': 'True'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'credential_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'team': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'credentials'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.Team']", 'blank': 'True', 'null': 'True'}),
+ 'user': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'credentials'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': u"orm['auth.User']", 'blank': 'True', 'null': 'True'})
+ },
+ 'main.group': {
+ 'Meta': {'unique_together': "(('name', 'inventory'),)", 'object_name': 'Group'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'group_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'group\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'hosts': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'groups'", 'blank': 'True', 'to': "orm['main.Host']"}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'inventory': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'groups'", 'to': "orm['main.Inventory']"}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'}),
+ 'parents': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'children'", 'blank': 'True', 'to': "orm['main.Group']"}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'group_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'variable_data': ('django.db.models.fields.related.OneToOneField', [], {'related_name': "'group'", 'unique': 'True', 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.VariableData']", 'blank': 'True', 'null': 'True'})
+ },
+ 'main.host': {
+ 'Meta': {'unique_together': "(('name', 'inventory'),)", 'object_name': 'Host'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'host_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'host\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'inventory': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'hosts'", 'to': "orm['main.Inventory']"}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'host_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'variable_data': ('django.db.models.fields.related.OneToOneField', [], {'related_name': "'host'", 'unique': 'True', 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.VariableData']", 'blank': 'True', 'null': 'True'})
+ },
+ 'main.inventory': {
+ 'Meta': {'unique_together': "(('name', 'organization'),)", 'object_name': 'Inventory'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'inventory_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'inventory\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'organization': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'inventories'", 'to': "orm['main.Organization']"}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'inventory_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"})
+ },
+ 'main.job': {
+ 'Meta': {'object_name': 'Job'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'job_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'celery_task_id': ('django.db.models.fields.CharField', [], {'default': "''", 'max_length': '100', 'blank': 'True'}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'job\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'credential': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'jobs'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.Credential']"}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'hosts': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'jobs'", 'blank': 'True', 'through': u"orm['main.JobHostSummary']", 'to': "orm['main.Host']"}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'inventory': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'jobs'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.Inventory']"}),
+ 'job_template': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'jobs'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.JobTemplate']", 'blank': 'True', 'null': 'True'}),
+ 'job_type': ('django.db.models.fields.CharField', [], {'max_length': '64'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'project': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'jobs'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['main.Project']"}),
+ 'result_stderr': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'result_stdout': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'result_traceback': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ 'status': ('django.db.models.fields.CharField', [], {'default': "'pending'", 'max_length': '20'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'job_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'user': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'jobs'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"})
+ },
+ 'main.jobevent': {
+ 'Meta': {'object_name': 'JobEvent'},
+ 'created': ('django.db.models.fields.DateTimeField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'event': ('django.db.models.fields.CharField', [], {'max_length': '100'}),
+ 'event_data': ('jsonfield.fields.JSONField', [], {'default': "''", 'blank': 'True'}),
+ 'host': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_events'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.Host']", 'blank': 'True', 'null': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'job': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_events'", 'to': "orm['main.Job']"})
+ },
+ u'main.jobhostsummary': {
+ 'Meta': {'ordering': "('-pk',)", 'unique_together': "[('job', 'host')]", 'object_name': 'JobHostSummary'},
+ 'changed': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
+ 'dark': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
+ 'failures': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
+ 'host': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_host_summaries'", 'to': "orm['main.Host']"}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'job': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_host_summaries'", 'to': "orm['main.Job']"}),
+ 'ok': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
+ 'processed': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'}),
+ 'skipped': ('django.db.models.fields.PositiveIntegerField', [], {'default': '0'})
+ },
+ 'main.jobtemplate': {
+ 'Meta': {'object_name': 'JobTemplate'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'jobtemplate_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'jobtemplate\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'credential': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_templates'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.Credential']", 'blank': 'True', 'null': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'inventory': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_templates'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': "orm['main.Inventory']", 'blank': 'True', 'null': 'True'}),
+ 'job_type': ('django.db.models.fields.CharField', [], {'max_length': '64'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'project': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_templates'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': u"orm['main.Project']", 'blank': 'True', 'null': 'True'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'jobtemplate_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'user': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'job_templates'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': u"orm['auth.User']", 'blank': 'True', 'null': 'True'})
+ },
+ 'main.organization': {
+ 'Meta': {'object_name': 'Organization'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'admins': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'admin_of_organizations'", 'blank': 'True', 'to': u"orm['auth.User']"}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'organization_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'organization\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'projects': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'organizations'", 'blank': 'True', 'to': u"orm['main.Project']"}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'organization_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'users': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'organizations'", 'blank': 'True', 'to': u"orm['auth.User']"})
+ },
+ 'main.permission': {
+ 'Meta': {'object_name': 'Permission'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'permission_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'permission\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'inventory': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'permissions'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.Inventory']"}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'}),
+ 'permission_type': ('django.db.models.fields.CharField', [], {'max_length': '64'}),
+ 'project': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'permissions'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['main.Project']"}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'permission_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'team': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'permissions'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.Team']"}),
+ 'user': ('django.db.models.fields.related.ForeignKey', [], {'blank': 'True', 'related_name': "'permissions'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"})
+ },
+ u'main.project': {
+ 'Meta': {'object_name': 'Project'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'project_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'project\', \'app_label\': u\'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'default_playbook': ('django.db.models.fields.CharField', [], {'max_length': '1024'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'local_repository': ('django.db.models.fields.CharField', [], {'max_length': '1024'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'scm_type': ('django.db.models.fields.CharField', [], {'max_length': '64'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'project_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"})
+ },
+ 'main.tag': {
+ 'Meta': {'object_name': 'Tag'},
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'})
+ },
+ 'main.team': {
+ 'Meta': {'object_name': 'Team'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'team_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'team\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
+ 'organization': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'teams'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.Organization']"}),
+ 'projects': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'teams'", 'blank': 'True', 'to': u"orm['main.Project']"}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'team_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
+ 'users': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'teams'", 'blank': 'True', 'to': u"orm['auth.User']"})
+ },
+ 'main.variabledata': {
+ 'Meta': {'object_name': 'VariableData'},
+ 'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
+ 'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'variabledata_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
+ 'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'variabledata\', \'app_label\': \'main\'}(class)s_created"', 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': u"orm['auth.User']"}),
+ 'creation_date': ('django.db.models.fields.DateField', [], {'auto_now_add': 'True', 'blank': 'True'}),
+ 'data': ('django.db.models.fields.TextField', [], {}),
+ 'description': ('django.db.models.fields.TextField', [], {'default': "''", 'blank': 'True'}),
+ u'id': ('django.db.models.fields.AutoField', [], {'primary_key': 'True'}),
+ 'name': ('django.db.models.fields.CharField', [], {'max_length': '512'}),
+ 'tags': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'variabledata_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"})
+ }
+ }
+
+ complete_apps = ['main']
\ No newline at end of file
diff --git a/lib/main/models/__init__.py b/lib/main/models/__init__.py
index e67b41bdbb..0002fb66c5 100644
--- a/lib/main/models/__init__.py
+++ b/lib/main/models/__init__.py
@@ -150,7 +150,7 @@ class PrimordialModel(models.Model):
abstract = True
description = models.TextField(blank=True, default='')
- created_by = models.ForeignKey('auth.User', on_delete=SET_NULL, null=True, related_name='%s(class)s_created') # not blank=False on purpose for admin!
+ created_by = models.ForeignKey('auth.User', on_delete=SET_NULL, null=True, related_name='%s(class)s_created', editable=False) # not blank=False on purpose for admin!
creation_date = models.DateField(auto_now_add=True)
tags = models.ManyToManyField('Tag', related_name='%(class)s_by_tag', blank=True)
audit_trail = models.ManyToManyField('AuditTrail', related_name='%(class)s_by_audit_trail', blank=True)
@@ -707,35 +707,51 @@ class Permission(CommonModelNameNotUnique):
# TODO: other job types (later)
-class LaunchJob(CommonModel):
+class JobTemplate(CommonModel):
'''
- A launch job is a definition for applying a project (with playbook) to an
- inventory source with a given credential.
+ A job template is a reusable job definition for applying a project (with
+ playbook) to an inventory source with a given credential.
'''
class Meta:
app_label = 'main'
- inventory = models.ForeignKey('Inventory', on_delete=SET_NULL, null=True, default=None, blank=True, related_name='launch_jobs')
- credential = models.ForeignKey('Credential', on_delete=SET_NULL, null=True, default=None, blank=True, related_name='launch_jobs')
- project = models.ForeignKey('Project', on_delete=SET_NULL, null=True, default=None, blank=True, related_name='launch_jobs')
- user = models.ForeignKey('auth.User', on_delete=SET_NULL, null=True, default=None, blank=True, related_name='launch_jobs')
-
- # JOB_TYPE_CHOICES are a subset of PERMISSION_TYPE_CHOICES
- job_type = models.CharField(max_length=64, choices=JOB_TYPE_CHOICES)
-
- def start(self):
- '''
- Create a new launch job status and start the task via celery.
- '''
- from lib.main.tasks import run_launch_job
- launch_job_status = self.launch_job_statuses.create(name='Launch Job Status %s' % now().isoformat())
- task_result = run_launch_job.delay(launch_job_status.pk)
- # The TaskMeta instance in the database isn't created until the worker
- # starts processing the task, so we can only store the task ID here.
- launch_job_status.celery_task_id = task_result.task_id
- launch_job_status.save(update_fields=['celery_task_id'])
- return launch_job_status
+ job_type = models.CharField(
+ max_length=64,
+ choices=JOB_TYPE_CHOICES,
+ )
+ inventory = models.ForeignKey(
+ 'Inventory',
+ related_name='job_templates',
+ blank=True,
+ null=True,
+ default=None,
+ on_delete=models.SET_NULL,
+ )
+ credential = models.ForeignKey(
+ 'Credential',
+ related_name='job_templates',
+ blank=True,
+ null=True,
+ default=None,
+ on_delete=models.SET_NULL,
+ )
+ project = models.ForeignKey(
+ 'Project',
+ related_name='job_templates',
+ blank=True,
+ null=True,
+ default=None,
+ on_delete=models.SET_NULL,
+ )
+ user = models.ForeignKey(
+ 'auth.User',
+ related_name='job_templates',
+ blank=True,
+ null=True,
+ default=None,
+ on_delete=models.SET_NULL,
+ )
# project has one default playbook but really should have a list of playbooks and flags ...
@@ -756,9 +772,11 @@ class LaunchJob(CommonModel):
# --list
# -- host