mirror of
https://github.com/ansible/awx.git
synced 2026-03-13 23:17:32 -02:30
Changed celery task reference to store the task_id only, capture traceback if task run fails, update admin to be able to run launch jobs.
This commit is contained in:
@@ -15,7 +15,13 @@
|
|||||||
# You should have received a copy of the GNU Affero General Public License
|
# 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/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
from django.conf.urls import *
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
from django.contrib.admin.util import unquote
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.core.urlresolvers import reverse
|
||||||
|
from django.http import HttpResponseRedirect
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
from lib.main.models import *
|
from lib.main.models import *
|
||||||
|
|
||||||
@@ -146,17 +152,55 @@ class PermissionAdmin(admin.ModelAdmin):
|
|||||||
|
|
||||||
class LaunchJobAdmin(admin.ModelAdmin):
|
class LaunchJobAdmin(admin.ModelAdmin):
|
||||||
|
|
||||||
list_display = ('name', 'description', 'active')
|
list_display = ('name', 'description', 'active', 'get_start_link_display',
|
||||||
|
'get_statuses_link_display')
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
(None, {'fields': ('name', 'active', 'created_by', 'description')}),
|
(None, {'fields': ('name', 'active', 'created_by', 'description',
|
||||||
|
'get_start_link_display', 'get_statuses_link_display')}),
|
||||||
(_('Job Parameters'), {'fields': ('inventory', 'project', 'credential',
|
(_('Job Parameters'), {'fields': ('inventory', 'project', 'credential',
|
||||||
'user', 'job_type')}),
|
'user', 'job_type')}),
|
||||||
(_('Tags'), {'fields': ('tags',)}),
|
(_('Tags'), {'fields': ('tags',)}),
|
||||||
(_('Audit Trail'), {'fields': ('creation_date', 'audit_trail',)}),
|
(_('Audit Trail'), {'fields': ('creation_date', 'audit_trail',)}),
|
||||||
)
|
)
|
||||||
readonly_fields = ('creation_date', 'audit_trail')
|
readonly_fields = ('creation_date', 'audit_trail', 'get_start_link_display',
|
||||||
|
'get_statuses_link_display')
|
||||||
filter_horizontal = ('tags',)
|
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 '<a href="%s">Run Job</a>' % 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 '<a href="%s">View Logs</a>' % 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,),
|
||||||
|
current_app=self.admin_site.name)
|
||||||
|
messages.success(request, '%s has been started.' % ljs)
|
||||||
|
return HttpResponseRedirect(status_url)
|
||||||
|
|
||||||
class LaunchJobStatusEventInline(admin.StackedInline):
|
class LaunchJobStatusEventInline(admin.StackedInline):
|
||||||
|
|
||||||
model = LaunchJobStatusEvent
|
model = LaunchJobStatusEvent
|
||||||
@@ -165,12 +209,23 @@ class LaunchJobStatusEventInline(admin.StackedInline):
|
|||||||
fields = ('created', 'event', 'event_data')
|
fields = ('created', 'event', 'event_data')
|
||||||
readonly_fields = ('created', 'event', 'event_data')
|
readonly_fields = ('created', 'event', 'event_data')
|
||||||
|
|
||||||
|
def has_add_permission(self, request):
|
||||||
|
return False
|
||||||
|
|
||||||
class LaunchJobStatusAdmin(admin.ModelAdmin):
|
class LaunchJobStatusAdmin(admin.ModelAdmin):
|
||||||
|
|
||||||
list_display = ('name', 'description', 'active', 'status')
|
list_display = ('name', 'launch_job', 'status')
|
||||||
|
fields = ('name', 'launch_job', 'status', 'result_stdout', 'result_stderr',
|
||||||
|
'result_traceback', 'celery_task_id', 'tags', 'created_by')
|
||||||
|
readonly_fields = ('name', 'description', 'status', 'launch_job',
|
||||||
|
'result_stdout', 'result_stderr', 'result_traceback',
|
||||||
|
'celery_task_id', 'created_by', 'tags', 'audit_trail', 'active')
|
||||||
filter_horizontal = ('tags',)
|
filter_horizontal = ('tags',)
|
||||||
inlines = [LaunchJobStatusEventInline]
|
inlines = [LaunchJobStatusEventInline]
|
||||||
|
|
||||||
|
def has_add_permission(self, request):
|
||||||
|
return False
|
||||||
|
|
||||||
# FIXME: Add the rest of the models...
|
# FIXME: Add the rest of the models...
|
||||||
|
|
||||||
admin.site.register(Organization, OrganizationAdmin)
|
admin.site.register(Organization, OrganizationAdmin)
|
||||||
|
|||||||
271
lib/main/migrations/0010_changes.py
Normal file
271
lib/main/migrations/0010_changes.py
Normal file
@@ -0,0 +1,271 @@
|
|||||||
|
# -*- 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):
|
||||||
|
# Adding field 'LaunchJobStatusEvent.host'
|
||||||
|
db.add_column(u'main_launchjobstatusevent', '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),
|
||||||
|
keep_default=False)
|
||||||
|
|
||||||
|
# Deleting field 'LaunchJobStatus.celery_task'
|
||||||
|
db.delete_column(u'main_launchjobstatus', 'celery_task_id')
|
||||||
|
|
||||||
|
# Adding field 'LaunchJobStatus.result_traceback'
|
||||||
|
db.add_column(u'main_launchjobstatus', 'result_traceback',
|
||||||
|
self.gf('django.db.models.fields.TextField')(default='', blank=True),
|
||||||
|
keep_default=False)
|
||||||
|
|
||||||
|
# Adding field 'LaunchJobStatus.celery_task_id'
|
||||||
|
db.add_column(u'main_launchjobstatus', 'celery_task_id',
|
||||||
|
self.gf('django.db.models.fields.CharField')(default='', max_length=100, blank=True),
|
||||||
|
keep_default=False)
|
||||||
|
|
||||||
|
|
||||||
|
def backwards(self, orm):
|
||||||
|
# Deleting field 'LaunchJobStatusEvent.host'
|
||||||
|
db.delete_column(u'main_launchjobstatusevent', 'host_id')
|
||||||
|
|
||||||
|
# Adding field 'LaunchJobStatus.celery_task'
|
||||||
|
db.add_column(u'main_launchjobstatus', 'celery_task',
|
||||||
|
self.gf('django.db.models.fields.related.ForeignKey')(related_name='launch_job_statuses', on_delete=models.SET_NULL, default=None, to=orm['djcelery.TaskMeta'], blank=True, null=True),
|
||||||
|
keep_default=False)
|
||||||
|
|
||||||
|
# Deleting field 'LaunchJobStatus.result_traceback'
|
||||||
|
db.delete_column(u'main_launchjobstatus', 'result_traceback')
|
||||||
|
|
||||||
|
# Deleting field 'LaunchJobStatus.celery_task_id'
|
||||||
|
db.delete_column(u'main_launchjobstatus', 'celery_task_id')
|
||||||
|
|
||||||
|
|
||||||
|
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.launchjob': {
|
||||||
|
'Meta': {'object_name': 'LaunchJob'},
|
||||||
|
'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
|
||||||
|
'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'launchjob_by_audit_trail'", 'blank': 'True', 'to': "orm['main.AuditTrail']"}),
|
||||||
|
'created_by': ('django.db.models.fields.related.ForeignKey', [], {'related_name': '"{\'class\': \'launchjob\', \'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': "'launch_jobs'", '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': "'launch_jobs'", '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': "'launch_jobs'", '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': "'launchjob_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"}),
|
||||||
|
'user': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'launch_jobs'", 'on_delete': 'models.SET_NULL', 'default': 'None', 'to': u"orm['auth.User']", 'blank': 'True', 'null': 'True'})
|
||||||
|
},
|
||||||
|
'main.launchjobstatus': {
|
||||||
|
'Meta': {'object_name': 'LaunchJobStatus'},
|
||||||
|
'active': ('django.db.models.fields.BooleanField', [], {'default': 'True'}),
|
||||||
|
'audit_trail': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'launchjobstatus_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\': \'launchjobstatus\', \'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'}),
|
||||||
|
'launch_job': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'launch_job_statuses'", 'null': 'True', 'on_delete': 'models.SET_NULL', 'to': "orm['main.LaunchJob']"}),
|
||||||
|
'name': ('django.db.models.fields.CharField', [], {'unique': 'True', 'max_length': '512'}),
|
||||||
|
'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': "'launchjobstatus_by_tag'", 'blank': 'True', 'to': "orm['main.Tag']"})
|
||||||
|
},
|
||||||
|
'main.launchjobstatusevent': {
|
||||||
|
'Meta': {'object_name': 'LaunchJobStatusEvent'},
|
||||||
|
'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': "'launch_job_status_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'}),
|
||||||
|
'launch_job_status': ('django.db.models.fields.related.ForeignKey', [], {'related_name': "'launch_job_status_events'", 'to': "orm['main.LaunchJobStatus']"})
|
||||||
|
},
|
||||||
|
'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'}),
|
||||||
|
'inventories': ('django.db.models.fields.related.ManyToManyField', [], {'symmetrical': 'False', 'related_name': "'projects'", 'blank': 'True', 'to': "orm['main.Inventory']"}),
|
||||||
|
'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']
|
||||||
@@ -716,11 +716,10 @@ class LaunchJob(CommonModel):
|
|||||||
from lib.main.tasks import run_launch_job
|
from lib.main.tasks import run_launch_job
|
||||||
launch_job_status = self.launch_job_statuses.create(name='Launch Job Status %s' % now().isoformat())
|
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)
|
task_result = run_launch_job.delay(launch_job_status.pk)
|
||||||
try:
|
# The TaskMeta instance in the database isn't created until the worker
|
||||||
launch_job_status.celery_task = TaskMeta.objects.get(task_id=task_result.task_id)
|
# starts processing the task, so we can only store the task ID here.
|
||||||
launch_job_status.save()
|
launch_job_status.celery_task_id = task_result.task_id
|
||||||
except TaskMeta.DoesNotExist:
|
launch_job_status.save(update_fields=['celery_task_id'])
|
||||||
pass
|
|
||||||
return launch_job_status
|
return launch_job_status
|
||||||
|
|
||||||
# project has one default playbook but really should have a list of playbooks and flags ...
|
# project has one default playbook but really should have a list of playbooks and flags ...
|
||||||
@@ -730,10 +729,6 @@ class LaunchJob(CommonModel):
|
|||||||
#
|
#
|
||||||
# playbook in source control is already on the disk
|
# playbook in source control is already on the disk
|
||||||
|
|
||||||
# job_type:
|
|
||||||
# run, check -- enough for now, more initially
|
|
||||||
# if check, add "--check" to parameters
|
|
||||||
|
|
||||||
# we'll extend ansible core to have callback context like
|
# we'll extend ansible core to have callback context like
|
||||||
# self.context.playbook
|
# self.context.playbook
|
||||||
# self.context.runner
|
# self.context.runner
|
||||||
@@ -756,17 +751,29 @@ class LaunchJobStatus(CommonModel):
|
|||||||
('running', _('Running')),
|
('running', _('Running')),
|
||||||
('successful', _('Successful')),
|
('successful', _('Successful')),
|
||||||
('failed', _('Failed')),
|
('failed', _('Failed')),
|
||||||
|
('error', _('Error')),
|
||||||
]
|
]
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
app_label = 'main'
|
app_label = 'main'
|
||||||
verbose_name_plural = _('launch job statuses')
|
verbose_name_plural = _('launch job statuses')
|
||||||
|
|
||||||
launch_job = models.ForeignKey('LaunchJob', null=True, on_delete=SET_NULL, related_name='launch_job_statuses')
|
launch_job = models.ForeignKey('LaunchJob', null=True, on_delete=SET_NULL, related_name='launch_job_statuses')
|
||||||
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='pending')
|
status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='pending')
|
||||||
result_stdout = models.TextField(blank=True, default='')
|
result_stdout = models.TextField(blank=True, default='')
|
||||||
result_stderr = models.TextField(blank=True, default='')
|
result_stderr = models.TextField(blank=True, default='')
|
||||||
celery_task = models.ForeignKey('djcelery.TaskMeta', related_name='launch_job_statuses', blank=True, null=True, default=None, on_delete=SET_NULL)
|
result_traceback = models.TextField(blank=True, default='')
|
||||||
|
celery_task_id = models.CharField(max_length=100, blank=True, default='', editable=False)
|
||||||
|
#hosts = models.ManyToManyField('Host', blank=True, related_name='launch_job_statuses')
|
||||||
|
# FIXME: Connect hosts based on inventory.
|
||||||
|
|
||||||
|
@property
|
||||||
|
def celery_task(self):
|
||||||
|
try:
|
||||||
|
if self.celery_task_id:
|
||||||
|
return TaskMeta.objects.get(task_id=self.celery_task_id)
|
||||||
|
except TaskMeta.DoesNotExist:
|
||||||
|
pass
|
||||||
|
|
||||||
class LaunchJobStatusEvent(models.Model):
|
class LaunchJobStatusEvent(models.Model):
|
||||||
'''
|
'''
|
||||||
@@ -801,5 +808,8 @@ class LaunchJobStatusEvent(models.Model):
|
|||||||
created = models.DateTimeField(auto_now_add=True)
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
event = models.CharField(max_length=100, choices=EVENT_TYPES)
|
event = models.CharField(max_length=100, choices=EVENT_TYPES)
|
||||||
event_data = JSONField(blank=True, default='')
|
event_data = JSONField(blank=True, default='')
|
||||||
|
host = models.ForeignKey('Host', blank=True, null=True, default=None, on_delete=SET_NULL, related_name='launch_job_status_events')
|
||||||
|
|
||||||
|
# FIXME: Connect host based on event_data.
|
||||||
|
|
||||||
# TODO: reporting (MPD)
|
# TODO: reporting (MPD)
|
||||||
|
|||||||
@@ -17,53 +17,55 @@
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
import subprocess
|
import subprocess
|
||||||
|
import traceback
|
||||||
from celery import task
|
from celery import task
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from lib.main.models import *
|
from lib.main.models import *
|
||||||
|
|
||||||
@task(name='run_launch_job')
|
@task(name='run_launch_job')
|
||||||
def run_launch_job(launch_job_status_pk):
|
def run_launch_job(launch_job_status_pk):
|
||||||
|
|
||||||
launch_job_status = LaunchJobStatus.objects.get(pk=launch_job_status_pk)
|
launch_job_status = LaunchJobStatus.objects.get(pk=launch_job_status_pk)
|
||||||
launch_job_status.status = 'running'
|
launch_job_status.status = 'running'
|
||||||
launch_job_status.save()
|
launch_job_status.save()
|
||||||
launch_job = launch_job_status.launch_job
|
launch_job = launch_job_status.launch_job
|
||||||
|
|
||||||
plugin_dir = os.path.abspath(os.path.join(os.path.dirname(__file__), '..',
|
|
||||||
'plugins', 'callback'))
|
|
||||||
inventory_script = os.path.abspath(os.path.join(os.path.dirname(__file__),
|
|
||||||
'management', 'commands',
|
|
||||||
'acom_inventory.py'))
|
|
||||||
callback_script = os.path.abspath(os.path.join(os.path.dirname(__file__),
|
|
||||||
'management', 'commands',
|
|
||||||
'acom_callback_event.py'))
|
|
||||||
env = dict(os.environ.items())
|
|
||||||
env['ACOM_LAUNCH_JOB_STATUS_ID'] = str(launch_job_status.pk)
|
|
||||||
env['ACOM_INVENTORY_ID'] = str(launch_job.inventory.pk)
|
|
||||||
env['ANSIBLE_CALLBACK_PLUGINS'] = plugin_dir
|
|
||||||
env['ACOM_CALLBACK_EVENT_SCRIPT'] = callback_script
|
|
||||||
|
|
||||||
if hasattr(settings, 'ANSIBLE_TRANSPORT'):
|
|
||||||
env['ANSIBLE_TRANSPORT'] = getattr(settings, 'ANSIBLE_TRANSPORT')
|
|
||||||
|
|
||||||
playbook = launch_job.project.default_playbook
|
|
||||||
cmdline = ['ansible-playbook', '-i', inventory_script]
|
|
||||||
if launch_job.job_type == 'check':
|
|
||||||
cmdline.append('--check')
|
|
||||||
cmdline.append(playbook)
|
|
||||||
|
|
||||||
# FIXME: How to cancel/interrupt job? (not that important for now)
|
try:
|
||||||
proc = subprocess.Popen(cmdline, stdout=subprocess.PIPE,
|
status, stdout, stderr, tb = 'error', '', '', ''
|
||||||
stderr=subprocess.PIPE, env=env)
|
plugin_dir = os.path.abspath(os.path.join(os.path.dirname(__file__), '..',
|
||||||
stdout, stderr = proc.communicate()
|
'plugins', 'callback'))
|
||||||
|
inventory_script = os.path.abspath(os.path.join(os.path.dirname(__file__),
|
||||||
|
'management', 'commands',
|
||||||
|
'acom_inventory.py'))
|
||||||
|
callback_script = os.path.abspath(os.path.join(os.path.dirname(__file__),
|
||||||
|
'management', 'commands',
|
||||||
|
'acom_callback_event.py'))
|
||||||
|
env = dict(os.environ.items())
|
||||||
|
env['ACOM_LAUNCH_JOB_STATUS_ID'] = str(launch_job_status.pk)
|
||||||
|
env['ACOM_INVENTORY_ID'] = str(launch_job.inventory.pk)
|
||||||
|
env['ANSIBLE_CALLBACK_PLUGINS'] = plugin_dir
|
||||||
|
env['ACOM_CALLBACK_EVENT_SCRIPT'] = callback_script
|
||||||
|
|
||||||
|
if hasattr(settings, 'ANSIBLE_TRANSPORT'):
|
||||||
|
env['ANSIBLE_TRANSPORT'] = getattr(settings, 'ANSIBLE_TRANSPORT')
|
||||||
|
|
||||||
|
playbook = launch_job.project.default_playbook
|
||||||
|
cmdline = ['ansible-playbook', '-i', inventory_script]
|
||||||
|
if launch_job.job_type == 'check':
|
||||||
|
cmdline.append('--check')
|
||||||
|
cmdline.append(playbook)
|
||||||
|
|
||||||
|
# FIXME: How to cancel/interrupt job? (not that important for now)
|
||||||
|
proc = subprocess.Popen(cmdline, stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.PIPE, env=env)
|
||||||
|
stdout, stderr = proc.communicate()
|
||||||
|
status = 'successful' if proc.returncode == 0 else 'failed'
|
||||||
|
except Exception:
|
||||||
|
tb = traceback.format_exc()
|
||||||
|
|
||||||
# Reload from database before updating/saving.
|
# Reload from database before updating/saving.
|
||||||
launch_job_status = LaunchJobStatus.objects.get(pk=launch_job_status_pk)
|
launch_job_status = LaunchJobStatus.objects.get(pk=launch_job_status_pk)
|
||||||
if proc.returncode == 0:
|
launch_job_status.status = status
|
||||||
launch_job_status.status = 'successful'
|
|
||||||
else:
|
|
||||||
launch_job_status.status = 'failed'
|
|
||||||
|
|
||||||
launch_job_status.result_stdout = stdout
|
launch_job_status.result_stdout = stdout
|
||||||
launch_job_status.result_stderr = stderr
|
launch_job_status.result_stderr = stderr
|
||||||
|
launch_job_status.result_traceback = tb
|
||||||
launch_job_status.save()
|
launch_job_status.save()
|
||||||
|
|||||||
Reference in New Issue
Block a user