models.py 61 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688
  1. #
  2. # ex:ts=4:sw=4:sts=4:et
  3. # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
  4. #
  5. # BitBake Toaster Implementation
  6. #
  7. # Copyright (C) 2013 Intel Corporation
  8. #
  9. # This program is free software; you can redistribute it and/or modify
  10. # it under the terms of the GNU General Public License version 2 as
  11. # published by the Free Software Foundation.
  12. #
  13. # This program is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU General Public License along
  19. # with this program; if not, write to the Free Software Foundation, Inc.,
  20. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. from __future__ import unicode_literals
  22. from django.db import models, IntegrityError
  23. from django.db.models import F, Q, Sum, Count
  24. from django.utils import timezone
  25. from django.utils.encoding import force_bytes
  26. from django.core.urlresolvers import reverse
  27. from django.core import validators
  28. from django.conf import settings
  29. import django.db.models.signals
  30. import sys
  31. import os.path
  32. import re
  33. import itertools
  34. import logging
  35. logger = logging.getLogger("toaster")
  36. if 'sqlite' in settings.DATABASES['default']['ENGINE']:
  37. from django.db import transaction, OperationalError
  38. from time import sleep
  39. _base_save = models.Model.save
  40. def save(self, *args, **kwargs):
  41. while True:
  42. try:
  43. with transaction.atomic():
  44. return _base_save(self, *args, **kwargs)
  45. except OperationalError as err:
  46. if 'database is locked' in str(err):
  47. logger.warning("%s, model: %s, args: %s, kwargs: %s",
  48. err, self.__class__, args, kwargs)
  49. sleep(0.5)
  50. continue
  51. raise
  52. models.Model.save = save
  53. # HACK: Monkey patch Django to fix 'database is locked' issue
  54. from django.db.models.query import QuerySet
  55. _base_insert = QuerySet._insert
  56. def _insert(self, *args, **kwargs):
  57. with transaction.atomic(using=self.db, savepoint=False):
  58. return _base_insert(self, *args, **kwargs)
  59. QuerySet._insert = _insert
  60. from django.utils import six
  61. def _create_object_from_params(self, lookup, params):
  62. """
  63. Tries to create an object using passed params.
  64. Used by get_or_create and update_or_create
  65. """
  66. try:
  67. obj = self.create(**params)
  68. return obj, True
  69. except IntegrityError:
  70. exc_info = sys.exc_info()
  71. try:
  72. return self.get(**lookup), False
  73. except self.model.DoesNotExist:
  74. pass
  75. six.reraise(*exc_info)
  76. QuerySet._create_object_from_params = _create_object_from_params
  77. # end of HACK
  78. class GitURLValidator(validators.URLValidator):
  79. import re
  80. regex = re.compile(
  81. r'^(?:ssh|git|http|ftp)s?://' # http:// or https://
  82. r'(?:(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+(?:[A-Z]{2,6}\.?|[A-Z0-9-]{2,}\.?)|' # domain...
  83. r'localhost|' # localhost...
  84. r'\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}|' # ...or ipv4
  85. r'\[?[A-F0-9]*:[A-F0-9:]+\]?)' # ...or ipv6
  86. r'(?::\d+)?' # optional port
  87. r'(?:/?|[/?]\S+)$', re.IGNORECASE)
  88. def GitURLField(**kwargs):
  89. r = models.URLField(**kwargs)
  90. for i in range(len(r.validators)):
  91. if isinstance(r.validators[i], validators.URLValidator):
  92. r.validators[i] = GitURLValidator()
  93. return r
  94. class ToasterSetting(models.Model):
  95. name = models.CharField(max_length=63)
  96. helptext = models.TextField()
  97. value = models.CharField(max_length=255)
  98. def __unicode__(self):
  99. return "Setting %s = %s" % (self.name, self.value)
  100. class ProjectManager(models.Manager):
  101. def create_project(self, name, release):
  102. if release is not None:
  103. prj = self.model(name=name,
  104. bitbake_version=release.bitbake_version,
  105. release=release)
  106. else:
  107. prj = self.model(name=name,
  108. bitbake_version=None,
  109. release=None)
  110. prj.save()
  111. for defaultconf in ToasterSetting.objects.filter(
  112. name__startswith="DEFCONF_"):
  113. name = defaultconf.name[8:]
  114. ProjectVariable.objects.create(project=prj,
  115. name=name,
  116. value=defaultconf.value)
  117. if release is None:
  118. return prj
  119. for rdl in release.releasedefaultlayer_set.all():
  120. lv = Layer_Version.objects.filter(
  121. layer__name=rdl.layer_name,
  122. release=release).first()
  123. if lv:
  124. ProjectLayer.objects.create(project=prj,
  125. layercommit=lv,
  126. optional=False)
  127. else:
  128. logger.warning("Default project layer %s not found" %
  129. rdl.layer_name)
  130. return prj
  131. # return single object with is_default = True
  132. def get_or_create_default_project(self):
  133. projects = super(ProjectManager, self).filter(is_default=True)
  134. if len(projects) > 1:
  135. raise Exception('Inconsistent project data: multiple ' +
  136. 'default projects (i.e. with is_default=True)')
  137. elif len(projects) < 1:
  138. options = {
  139. 'name': 'Command line builds',
  140. 'short_description':
  141. 'Project for builds started outside Toaster',
  142. 'is_default': True
  143. }
  144. project = Project.objects.create(**options)
  145. project.save()
  146. return project
  147. else:
  148. return projects[0]
  149. class Project(models.Model):
  150. search_allowed_fields = ['name', 'short_description', 'release__name', 'release__branch_name']
  151. name = models.CharField(max_length=100)
  152. short_description = models.CharField(max_length=50, blank=True)
  153. bitbake_version = models.ForeignKey('BitbakeVersion', null=True)
  154. release = models.ForeignKey("Release", null=True)
  155. created = models.DateTimeField(auto_now_add = True)
  156. updated = models.DateTimeField(auto_now = True)
  157. # This is a horrible hack; since Toaster has no "User" model available when
  158. # running in interactive mode, we can't reference the field here directly
  159. # Instead, we keep a possible null reference to the User id, as not to force
  160. # hard links to possibly missing models
  161. user_id = models.IntegerField(null = True)
  162. objects = ProjectManager()
  163. # set to True for the project which is the default container
  164. # for builds initiated by the command line etc.
  165. is_default = models.BooleanField(default = False)
  166. def __unicode__(self):
  167. return "%s (Release %s, BBV %s)" % (self.name, self.release, self.bitbake_version)
  168. def get_current_machine_name(self):
  169. try:
  170. return self.projectvariable_set.get(name="MACHINE").value
  171. except (ProjectVariable.DoesNotExist,IndexError):
  172. return None;
  173. def get_number_of_builds(self):
  174. """Return the number of builds which have ended"""
  175. return self.build_set.exclude(
  176. Q(outcome=Build.IN_PROGRESS) |
  177. Q(outcome=Build.CANCELLED)
  178. ).count()
  179. def get_last_build_id(self):
  180. try:
  181. return Build.objects.filter( project = self.id ).order_by('-completed_on')[0].id
  182. except (Build.DoesNotExist,IndexError):
  183. return( -1 )
  184. def get_last_outcome(self):
  185. build_id = self.get_last_build_id
  186. if (-1 == build_id):
  187. return( "" )
  188. try:
  189. return Build.objects.filter( id = self.get_last_build_id )[ 0 ].outcome
  190. except (Build.DoesNotExist,IndexError):
  191. return( "not_found" )
  192. def get_last_target(self):
  193. build_id = self.get_last_build_id
  194. if (-1 == build_id):
  195. return( "" )
  196. try:
  197. return Target.objects.filter(build = build_id)[0].target
  198. except (Target.DoesNotExist,IndexError):
  199. return( "not_found" )
  200. def get_last_errors(self):
  201. build_id = self.get_last_build_id
  202. if (-1 == build_id):
  203. return( 0 )
  204. try:
  205. return Build.objects.filter(id = build_id)[ 0 ].errors.count()
  206. except (Build.DoesNotExist,IndexError):
  207. return( "not_found" )
  208. def get_last_warnings(self):
  209. build_id = self.get_last_build_id
  210. if (-1 == build_id):
  211. return( 0 )
  212. try:
  213. return Build.objects.filter(id = build_id)[ 0 ].warnings.count()
  214. except (Build.DoesNotExist,IndexError):
  215. return( "not_found" )
  216. def get_last_build_extensions(self):
  217. """
  218. Get list of file name extensions for images produced by the most
  219. recent build
  220. """
  221. last_build = Build.objects.get(pk = self.get_last_build_id())
  222. return last_build.get_image_file_extensions()
  223. def get_last_imgfiles(self):
  224. build_id = self.get_last_build_id
  225. if (-1 == build_id):
  226. return( "" )
  227. try:
  228. return Variable.objects.filter(build = build_id, variable_name = "IMAGE_FSTYPES")[ 0 ].variable_value
  229. except (Variable.DoesNotExist,IndexError):
  230. return( "not_found" )
  231. def get_all_compatible_layer_versions(self):
  232. """ Returns Queryset of all Layer_Versions which are compatible with
  233. this project"""
  234. queryset = None
  235. # guard on release, as it can be null
  236. if self.release:
  237. queryset = Layer_Version.objects.filter(
  238. (Q(release=self.release) &
  239. Q(build=None) &
  240. Q(project=None)) |
  241. Q(project=self))
  242. else:
  243. queryset = Layer_Version.objects.none()
  244. return queryset
  245. def get_project_layer_versions(self, pk=False):
  246. """ Returns the Layer_Versions currently added to this project """
  247. layer_versions = self.projectlayer_set.all().values_list('layercommit',
  248. flat=True)
  249. if pk is False:
  250. return Layer_Version.objects.filter(pk__in=layer_versions)
  251. else:
  252. return layer_versions
  253. def get_available_machines(self):
  254. """ Returns QuerySet of all Machines which are provided by the
  255. Layers currently added to the Project """
  256. queryset = Machine.objects.filter(
  257. layer_version__in=self.get_project_layer_versions())
  258. return queryset
  259. def get_all_compatible_machines(self):
  260. """ Returns QuerySet of all the compatible machines available to the
  261. project including ones from Layers not currently added """
  262. queryset = Machine.objects.filter(
  263. layer_version__in=self.get_all_compatible_layer_versions())
  264. return queryset
  265. def get_available_recipes(self):
  266. """ Returns QuerySet of all the recipes that are provided by layers
  267. added to this project """
  268. queryset = Recipe.objects.filter(
  269. layer_version__in=self.get_project_layer_versions())
  270. return queryset
  271. def get_all_compatible_recipes(self):
  272. """ Returns QuerySet of all the compatible Recipes available to the
  273. project including ones from Layers not currently added """
  274. queryset = Recipe.objects.filter(
  275. layer_version__in=self.get_all_compatible_layer_versions()).exclude(name__exact='')
  276. return queryset
  277. def schedule_build(self):
  278. from bldcontrol.models import BuildRequest, BRTarget, BRLayer, BRVariable, BRBitbake
  279. br = BuildRequest.objects.create(project = self)
  280. try:
  281. BRBitbake.objects.create(req = br,
  282. giturl = self.bitbake_version.giturl,
  283. commit = self.bitbake_version.branch,
  284. dirpath = self.bitbake_version.dirpath)
  285. for l in self.projectlayer_set.all().order_by("pk"):
  286. commit = l.layercommit.get_vcs_reference()
  287. print("ii Building layer ", l.layercommit.layer.name, " at vcs point ", commit)
  288. BRLayer.objects.create(req = br, name = l.layercommit.layer.name, giturl = l.layercommit.layer.vcs_url, commit = commit, dirpath = l.layercommit.dirpath, layer_version=l.layercommit)
  289. br.state = BuildRequest.REQ_QUEUED
  290. now = timezone.now()
  291. br.build = Build.objects.create(project = self,
  292. completed_on=now,
  293. started_on=now,
  294. )
  295. for t in self.projecttarget_set.all():
  296. BRTarget.objects.create(req = br, target = t.target, task = t.task)
  297. Target.objects.create(build = br.build, target = t.target, task = t.task)
  298. for v in self.projectvariable_set.all():
  299. BRVariable.objects.create(req = br, name = v.name, value = v.value)
  300. try:
  301. br.build.machine = self.projectvariable_set.get(name = 'MACHINE').value
  302. br.build.save()
  303. except ProjectVariable.DoesNotExist:
  304. pass
  305. br.save()
  306. except Exception:
  307. # revert the build request creation since we're not done cleanly
  308. br.delete()
  309. raise
  310. return br
  311. class Build(models.Model):
  312. SUCCEEDED = 0
  313. FAILED = 1
  314. IN_PROGRESS = 2
  315. CANCELLED = 3
  316. BUILD_OUTCOME = (
  317. (SUCCEEDED, 'Succeeded'),
  318. (FAILED, 'Failed'),
  319. (IN_PROGRESS, 'In Progress'),
  320. (CANCELLED, 'Cancelled'),
  321. )
  322. search_allowed_fields = ['machine', 'cooker_log_path', "target__target", "target__target_image_file__file_name"]
  323. project = models.ForeignKey(Project) # must have a project
  324. machine = models.CharField(max_length=100)
  325. distro = models.CharField(max_length=100)
  326. distro_version = models.CharField(max_length=100)
  327. started_on = models.DateTimeField()
  328. completed_on = models.DateTimeField()
  329. outcome = models.IntegerField(choices=BUILD_OUTCOME, default=IN_PROGRESS)
  330. cooker_log_path = models.CharField(max_length=500)
  331. build_name = models.CharField(max_length=100)
  332. bitbake_version = models.CharField(max_length=50)
  333. @staticmethod
  334. def get_recent(project=None):
  335. """
  336. Return recent builds as a list; if project is set, only return
  337. builds for that project
  338. """
  339. builds = Build.objects.all()
  340. if project:
  341. builds = builds.filter(project=project)
  342. finished_criteria = \
  343. Q(outcome=Build.SUCCEEDED) | \
  344. Q(outcome=Build.FAILED) | \
  345. Q(outcome=Build.CANCELLED)
  346. recent_builds = list(itertools.chain(
  347. builds.filter(outcome=Build.IN_PROGRESS).order_by("-started_on"),
  348. builds.filter(finished_criteria).order_by("-completed_on")[:3]
  349. ))
  350. # add percentage done property to each build; this is used
  351. # to show build progress in mrb_section.html
  352. for build in recent_builds:
  353. build.percentDone = build.completeper()
  354. build.outcomeText = build.get_outcome_text()
  355. return recent_builds
  356. def completeper(self):
  357. tf = Task.objects.filter(build = self)
  358. tfc = tf.count()
  359. if tfc > 0:
  360. completeper = tf.exclude(order__isnull=True).count()*100 // tfc
  361. else:
  362. completeper = 0
  363. return completeper
  364. def eta(self):
  365. eta = timezone.now()
  366. completeper = self.completeper()
  367. if self.completeper() > 0:
  368. eta += ((eta - self.started_on)*(100-completeper))/completeper
  369. return eta
  370. def has_images(self):
  371. """
  372. Returns True if at least one of the targets for this build has an
  373. image file associated with it, False otherwise
  374. """
  375. targets = Target.objects.filter(build_id=self.id)
  376. has_images = False
  377. for target in targets:
  378. if target.has_images():
  379. has_images = True
  380. break
  381. return has_images
  382. def has_image_recipes(self):
  383. """
  384. Returns True if a build has any targets which were built from
  385. image recipes.
  386. """
  387. image_recipes = self.get_image_recipes()
  388. return len(image_recipes) > 0
  389. def get_image_file_extensions(self):
  390. """
  391. Get string of file name extensions for images produced by this build;
  392. note that this is the actual list of extensions stored on Target objects
  393. for this build, and not the value of IMAGE_FSTYPES.
  394. Returns comma-separated string, e.g. "vmdk, ext4"
  395. """
  396. extensions = []
  397. targets = Target.objects.filter(build_id = self.id)
  398. for target in targets:
  399. if not target.is_image:
  400. continue
  401. target_image_files = Target_Image_File.objects.filter(
  402. target_id=target.id)
  403. for target_image_file in target_image_files:
  404. extensions.append(target_image_file.suffix)
  405. extensions = list(set(extensions))
  406. extensions.sort()
  407. return ', '.join(extensions)
  408. def get_image_fstypes(self):
  409. """
  410. Get the IMAGE_FSTYPES variable value for this build as a de-duplicated
  411. list of image file suffixes.
  412. """
  413. image_fstypes = Variable.objects.get(
  414. build=self, variable_name='IMAGE_FSTYPES').variable_value
  415. return list(set(re.split(r' {1,}', image_fstypes)))
  416. def get_sorted_target_list(self):
  417. tgts = Target.objects.filter(build_id = self.id).order_by( 'target' );
  418. return( tgts );
  419. def get_recipes(self):
  420. """
  421. Get the recipes related to this build;
  422. note that the related layer versions and layers are also prefetched
  423. by this query, as this queryset can be sorted by these objects in the
  424. build recipes view; prefetching them here removes the need
  425. for another query in that view
  426. """
  427. layer_versions = Layer_Version.objects.filter(build=self)
  428. criteria = Q(layer_version__id__in=layer_versions)
  429. return Recipe.objects.filter(criteria) \
  430. .select_related('layer_version', 'layer_version__layer')
  431. def get_image_recipes(self):
  432. """
  433. Returns a list of image Recipes (custom and built-in) related to this
  434. build, sorted by name; note that this has to be done in two steps, as
  435. there's no way to get all the custom image recipes and image recipes
  436. in one query
  437. """
  438. custom_image_recipes = self.get_custom_image_recipes()
  439. custom_image_recipe_names = custom_image_recipes.values_list('name', flat=True)
  440. not_custom_image_recipes = ~Q(name__in=custom_image_recipe_names) & \
  441. Q(is_image=True)
  442. built_image_recipes = self.get_recipes().filter(not_custom_image_recipes)
  443. # append to the custom image recipes and sort
  444. customisable_image_recipes = list(
  445. itertools.chain(custom_image_recipes, built_image_recipes)
  446. )
  447. return sorted(customisable_image_recipes, key=lambda recipe: recipe.name)
  448. def get_custom_image_recipes(self):
  449. """
  450. Returns a queryset of CustomImageRecipes related to this build,
  451. sorted by name
  452. """
  453. built_recipe_names = self.get_recipes().values_list('name', flat=True)
  454. criteria = Q(name__in=built_recipe_names) & Q(project=self.project)
  455. queryset = CustomImageRecipe.objects.filter(criteria).order_by('name')
  456. return queryset
  457. def get_outcome_text(self):
  458. return Build.BUILD_OUTCOME[int(self.outcome)][1]
  459. @property
  460. def failed_tasks(self):
  461. """ Get failed tasks for the build """
  462. tasks = self.task_build.all()
  463. return tasks.filter(order__gt=0, outcome=Task.OUTCOME_FAILED)
  464. @property
  465. def errors(self):
  466. return (self.logmessage_set.filter(level=LogMessage.ERROR) |
  467. self.logmessage_set.filter(level=LogMessage.EXCEPTION) |
  468. self.logmessage_set.filter(level=LogMessage.CRITICAL))
  469. @property
  470. def warnings(self):
  471. return self.logmessage_set.filter(level=LogMessage.WARNING)
  472. @property
  473. def timespent(self):
  474. return self.completed_on - self.started_on
  475. @property
  476. def timespent_seconds(self):
  477. return self.timespent.total_seconds()
  478. @property
  479. def target_labels(self):
  480. """
  481. Sorted (a-z) "target1:task, target2, target3" etc. string for all
  482. targets in this build
  483. """
  484. targets = self.target_set.all()
  485. target_labels = [target.target +
  486. (':' + target.task if target.task else '')
  487. for target in targets]
  488. target_labels.sort()
  489. return target_labels
  490. def get_buildrequest(self):
  491. buildrequest = None
  492. if hasattr(self, 'buildrequest'):
  493. buildrequest = self.buildrequest
  494. return buildrequest
  495. def is_queued(self):
  496. from bldcontrol.models import BuildRequest
  497. buildrequest = self.get_buildrequest()
  498. if buildrequest:
  499. return buildrequest.state == BuildRequest.REQ_QUEUED
  500. else:
  501. return False
  502. def is_cancelling(self):
  503. from bldcontrol.models import BuildRequest
  504. buildrequest = self.get_buildrequest()
  505. if buildrequest:
  506. return self.outcome == Build.IN_PROGRESS and \
  507. buildrequest.state == BuildRequest.REQ_CANCELLING
  508. else:
  509. return False
  510. def get_state(self):
  511. """
  512. Get the state of the build; one of 'Succeeded', 'Failed', 'In Progress',
  513. 'Cancelled' (Build outcomes); or 'Queued', 'Cancelling' (states
  514. dependent on the BuildRequest state).
  515. This works around the fact that we have BuildRequest states as well
  516. as Build states, but really we just want to know the state of the build.
  517. """
  518. if self.is_cancelling():
  519. return 'Cancelling';
  520. elif self.is_queued():
  521. return 'Queued'
  522. else:
  523. return self.get_outcome_text()
  524. def __str__(self):
  525. return "%d %s %s" % (self.id, self.project, ",".join([t.target for t in self.target_set.all()]))
  526. class ProjectTarget(models.Model):
  527. project = models.ForeignKey(Project)
  528. target = models.CharField(max_length=100)
  529. task = models.CharField(max_length=100, null=True)
  530. class Target(models.Model):
  531. search_allowed_fields = ['target', 'file_name']
  532. build = models.ForeignKey(Build)
  533. target = models.CharField(max_length=100)
  534. task = models.CharField(max_length=100, null=True)
  535. is_image = models.BooleanField(default = False)
  536. image_size = models.IntegerField(default=0)
  537. license_manifest_path = models.CharField(max_length=500, null=True)
  538. package_manifest_path = models.CharField(max_length=500, null=True)
  539. def package_count(self):
  540. return Target_Installed_Package.objects.filter(target_id__exact=self.id).count()
  541. def __unicode__(self):
  542. return self.target
  543. def get_similar_targets(self):
  544. """
  545. Get target sfor the same machine, task and target name
  546. (e.g. 'core-image-minimal') from a successful build for this project
  547. (but excluding this target).
  548. Note that we only look for targets built by this project because
  549. projects can have different configurations from each other, and put
  550. their artifacts in different directories.
  551. The possibility of error when retrieving candidate targets
  552. is minimised by the fact that bitbake will rebuild artifacts if MACHINE
  553. (or various other variables) change. In this case, there is no need to
  554. clone artifacts from another target, as those artifacts will have
  555. been re-generated for this target anyway.
  556. """
  557. query = ~Q(pk=self.pk) & \
  558. Q(target=self.target) & \
  559. Q(build__machine=self.build.machine) & \
  560. Q(build__outcome=Build.SUCCEEDED) & \
  561. Q(build__project=self.build.project)
  562. return Target.objects.filter(query)
  563. def get_similar_target_with_image_files(self):
  564. """
  565. Get the most recent similar target with Target_Image_Files associated
  566. with it, for the purpose of cloning those files onto this target.
  567. """
  568. similar_target = None
  569. candidates = self.get_similar_targets()
  570. if candidates.count() == 0:
  571. return similar_target
  572. task_subquery = Q(task=self.task)
  573. # we can look for a 'build' task if this task is a 'populate_sdk_ext'
  574. # task, as the latter also creates images; and vice versa; note that
  575. # 'build' targets can have their task set to '';
  576. # also note that 'populate_sdk' does not produce image files
  577. image_tasks = [
  578. '', # aka 'build'
  579. 'build',
  580. 'image',
  581. 'populate_sdk_ext'
  582. ]
  583. if self.task in image_tasks:
  584. task_subquery = Q(task__in=image_tasks)
  585. # annotate with the count of files, to exclude any targets which
  586. # don't have associated files
  587. candidates = candidates.annotate(num_files=Count('target_image_file'))
  588. query = task_subquery & Q(num_files__gt=0)
  589. candidates = candidates.filter(query)
  590. if candidates.count() > 0:
  591. candidates.order_by('build__completed_on')
  592. similar_target = candidates.last()
  593. return similar_target
  594. def get_similar_target_with_sdk_files(self):
  595. """
  596. Get the most recent similar target with TargetSDKFiles associated
  597. with it, for the purpose of cloning those files onto this target.
  598. """
  599. similar_target = None
  600. candidates = self.get_similar_targets()
  601. if candidates.count() == 0:
  602. return similar_target
  603. # annotate with the count of files, to exclude any targets which
  604. # don't have associated files
  605. candidates = candidates.annotate(num_files=Count('targetsdkfile'))
  606. query = Q(task=self.task) & Q(num_files__gt=0)
  607. candidates = candidates.filter(query)
  608. if candidates.count() > 0:
  609. candidates.order_by('build__completed_on')
  610. similar_target = candidates.last()
  611. return similar_target
  612. def clone_image_artifacts_from(self, target):
  613. """
  614. Make clones of the Target_Image_Files and TargetKernelFile objects
  615. associated with Target target, then associate them with this target.
  616. Note that for Target_Image_Files, we only want files from the previous
  617. build whose suffix matches one of the suffixes defined in this
  618. target's build's IMAGE_FSTYPES configuration variable. This prevents the
  619. Target_Image_File object for an ext4 image being associated with a
  620. target for a project which didn't produce an ext4 image (for example).
  621. Also sets the license_manifest_path and package_manifest_path
  622. of this target to the same path as that of target being cloned from, as
  623. the manifests are also build artifacts but are treated differently.
  624. """
  625. image_fstypes = self.build.get_image_fstypes()
  626. # filter out any image files whose suffixes aren't in the
  627. # IMAGE_FSTYPES suffixes variable for this target's build
  628. image_files = [target_image_file \
  629. for target_image_file in target.target_image_file_set.all() \
  630. if target_image_file.suffix in image_fstypes]
  631. for image_file in image_files:
  632. image_file.pk = None
  633. image_file.target = self
  634. image_file.save()
  635. kernel_files = target.targetkernelfile_set.all()
  636. for kernel_file in kernel_files:
  637. kernel_file.pk = None
  638. kernel_file.target = self
  639. kernel_file.save()
  640. self.license_manifest_path = target.license_manifest_path
  641. self.package_manifest_path = target.package_manifest_path
  642. self.save()
  643. def clone_sdk_artifacts_from(self, target):
  644. """
  645. Clone TargetSDKFile objects from target and associate them with this
  646. target.
  647. """
  648. sdk_files = target.targetsdkfile_set.all()
  649. for sdk_file in sdk_files:
  650. sdk_file.pk = None
  651. sdk_file.target = self
  652. sdk_file.save()
  653. def has_images(self):
  654. """
  655. Returns True if this target has one or more image files attached to it.
  656. """
  657. return self.target_image_file_set.all().count() > 0
  658. # kernel artifacts for a target: bzImage and modules*
  659. class TargetKernelFile(models.Model):
  660. target = models.ForeignKey(Target)
  661. file_name = models.FilePathField()
  662. file_size = models.IntegerField()
  663. @property
  664. def basename(self):
  665. return os.path.basename(self.file_name)
  666. # SDK artifacts for a target: sh and manifest files
  667. class TargetSDKFile(models.Model):
  668. target = models.ForeignKey(Target)
  669. file_name = models.FilePathField()
  670. file_size = models.IntegerField()
  671. @property
  672. def basename(self):
  673. return os.path.basename(self.file_name)
  674. class Target_Image_File(models.Model):
  675. # valid suffixes for image files produced by a build
  676. SUFFIXES = {
  677. 'btrfs', 'cpio', 'cpio.gz', 'cpio.lz4', 'cpio.lzma', 'cpio.xz',
  678. 'cramfs', 'elf', 'ext2', 'ext2.bz2', 'ext2.gz', 'ext2.lzma', 'ext4',
  679. 'ext4.gz', 'ext3', 'ext3.gz', 'hddimg', 'iso', 'jffs2', 'jffs2.sum',
  680. 'squashfs', 'squashfs-lzo', 'squashfs-xz', 'tar.bz2', 'tar.lz4',
  681. 'tar.xz', 'tartar.gz', 'ubi', 'ubifs', 'vmdk'
  682. }
  683. target = models.ForeignKey(Target)
  684. file_name = models.FilePathField(max_length=254)
  685. file_size = models.IntegerField()
  686. @property
  687. def suffix(self):
  688. """
  689. Suffix for image file, minus leading "."
  690. """
  691. for suffix in Target_Image_File.SUFFIXES:
  692. if self.file_name.endswith(suffix):
  693. return suffix
  694. filename, suffix = os.path.splitext(self.file_name)
  695. suffix = suffix.lstrip('.')
  696. return suffix
  697. class Target_File(models.Model):
  698. ITYPE_REGULAR = 1
  699. ITYPE_DIRECTORY = 2
  700. ITYPE_SYMLINK = 3
  701. ITYPE_SOCKET = 4
  702. ITYPE_FIFO = 5
  703. ITYPE_CHARACTER = 6
  704. ITYPE_BLOCK = 7
  705. ITYPES = ( (ITYPE_REGULAR ,'regular'),
  706. ( ITYPE_DIRECTORY ,'directory'),
  707. ( ITYPE_SYMLINK ,'symlink'),
  708. ( ITYPE_SOCKET ,'socket'),
  709. ( ITYPE_FIFO ,'fifo'),
  710. ( ITYPE_CHARACTER ,'character'),
  711. ( ITYPE_BLOCK ,'block'),
  712. )
  713. target = models.ForeignKey(Target)
  714. path = models.FilePathField()
  715. size = models.IntegerField()
  716. inodetype = models.IntegerField(choices = ITYPES)
  717. permission = models.CharField(max_length=16)
  718. owner = models.CharField(max_length=128)
  719. group = models.CharField(max_length=128)
  720. directory = models.ForeignKey('Target_File', related_name="directory_set", null=True)
  721. sym_target = models.ForeignKey('Target_File', related_name="symlink_set", null=True)
  722. class Task(models.Model):
  723. SSTATE_NA = 0
  724. SSTATE_MISS = 1
  725. SSTATE_FAILED = 2
  726. SSTATE_RESTORED = 3
  727. SSTATE_RESULT = (
  728. (SSTATE_NA, 'Not Applicable'), # For rest of tasks, but they still need checking.
  729. (SSTATE_MISS, 'File not in cache'), # the sstate object was not found
  730. (SSTATE_FAILED, 'Failed'), # there was a pkg, but the script failed
  731. (SSTATE_RESTORED, 'Succeeded'), # successfully restored
  732. )
  733. CODING_NA = 0
  734. CODING_PYTHON = 2
  735. CODING_SHELL = 3
  736. TASK_CODING = (
  737. (CODING_NA, 'N/A'),
  738. (CODING_PYTHON, 'Python'),
  739. (CODING_SHELL, 'Shell'),
  740. )
  741. OUTCOME_NA = -1
  742. OUTCOME_SUCCESS = 0
  743. OUTCOME_COVERED = 1
  744. OUTCOME_CACHED = 2
  745. OUTCOME_PREBUILT = 3
  746. OUTCOME_FAILED = 4
  747. OUTCOME_EMPTY = 5
  748. TASK_OUTCOME = (
  749. (OUTCOME_NA, 'Not Available'),
  750. (OUTCOME_SUCCESS, 'Succeeded'),
  751. (OUTCOME_COVERED, 'Covered'),
  752. (OUTCOME_CACHED, 'Cached'),
  753. (OUTCOME_PREBUILT, 'Prebuilt'),
  754. (OUTCOME_FAILED, 'Failed'),
  755. (OUTCOME_EMPTY, 'Empty'),
  756. )
  757. TASK_OUTCOME_HELP = (
  758. (OUTCOME_SUCCESS, 'This task successfully completed'),
  759. (OUTCOME_COVERED, 'This task did not run because its output is provided by another task'),
  760. (OUTCOME_CACHED, 'This task restored output from the sstate-cache directory or mirrors'),
  761. (OUTCOME_PREBUILT, 'This task did not run because its outcome was reused from a previous build'),
  762. (OUTCOME_FAILED, 'This task did not complete'),
  763. (OUTCOME_EMPTY, 'This task has no executable content'),
  764. (OUTCOME_NA, ''),
  765. )
  766. search_allowed_fields = [ "recipe__name", "recipe__version", "task_name", "logfile" ]
  767. def __init__(self, *args, **kwargs):
  768. super(Task, self).__init__(*args, **kwargs)
  769. try:
  770. self._helptext = HelpText.objects.get(key=self.task_name, area=HelpText.VARIABLE, build=self.build).text
  771. except HelpText.DoesNotExist:
  772. self._helptext = None
  773. def get_related_setscene(self):
  774. return Task.objects.filter(task_executed=True, build = self.build, recipe = self.recipe, task_name=self.task_name+"_setscene")
  775. def get_outcome_text(self):
  776. return Task.TASK_OUTCOME[int(self.outcome) + 1][1]
  777. def get_outcome_help(self):
  778. return Task.TASK_OUTCOME_HELP[int(self.outcome)][1]
  779. def get_sstate_text(self):
  780. if self.sstate_result==Task.SSTATE_NA:
  781. return ''
  782. else:
  783. return Task.SSTATE_RESULT[int(self.sstate_result)][1]
  784. def get_executed_display(self):
  785. if self.task_executed:
  786. return "Executed"
  787. return "Not Executed"
  788. def get_description(self):
  789. return self._helptext
  790. build = models.ForeignKey(Build, related_name='task_build')
  791. order = models.IntegerField(null=True)
  792. task_executed = models.BooleanField(default=False) # True means Executed, False means Not/Executed
  793. outcome = models.IntegerField(choices=TASK_OUTCOME, default=OUTCOME_NA)
  794. sstate_checksum = models.CharField(max_length=100, blank=True)
  795. path_to_sstate_obj = models.FilePathField(max_length=500, blank=True)
  796. recipe = models.ForeignKey('Recipe', related_name='tasks')
  797. task_name = models.CharField(max_length=100)
  798. source_url = models.FilePathField(max_length=255, blank=True)
  799. work_directory = models.FilePathField(max_length=255, blank=True)
  800. script_type = models.IntegerField(choices=TASK_CODING, default=CODING_NA)
  801. line_number = models.IntegerField(default=0)
  802. # start/end times
  803. started = models.DateTimeField(null=True)
  804. ended = models.DateTimeField(null=True)
  805. # in seconds; this is stored to enable sorting
  806. elapsed_time = models.DecimalField(max_digits=8, decimal_places=2, null=True)
  807. # in bytes; note that disk_io is stored to enable sorting
  808. disk_io = models.IntegerField(null=True)
  809. disk_io_read = models.IntegerField(null=True)
  810. disk_io_write = models.IntegerField(null=True)
  811. # in seconds
  812. cpu_time_user = models.DecimalField(max_digits=8, decimal_places=2, null=True)
  813. cpu_time_system = models.DecimalField(max_digits=8, decimal_places=2, null=True)
  814. sstate_result = models.IntegerField(choices=SSTATE_RESULT, default=SSTATE_NA)
  815. message = models.CharField(max_length=240)
  816. logfile = models.FilePathField(max_length=255, blank=True)
  817. outcome_text = property(get_outcome_text)
  818. sstate_text = property(get_sstate_text)
  819. def __unicode__(self):
  820. return "%d(%d) %s:%s" % (self.pk, self.build.pk, self.recipe.name, self.task_name)
  821. class Meta:
  822. ordering = ('order', 'recipe' ,)
  823. unique_together = ('build', 'recipe', 'task_name', )
  824. class Task_Dependency(models.Model):
  825. task = models.ForeignKey(Task, related_name='task_dependencies_task')
  826. depends_on = models.ForeignKey(Task, related_name='task_dependencies_depends')
  827. class Package(models.Model):
  828. search_allowed_fields = ['name', 'version', 'revision', 'recipe__name', 'recipe__version', 'recipe__license', 'recipe__layer_version__layer__name', 'recipe__layer_version__branch', 'recipe__layer_version__commit', 'recipe__layer_version__local_path', 'installed_name']
  829. build = models.ForeignKey('Build', null=True)
  830. recipe = models.ForeignKey('Recipe', null=True)
  831. name = models.CharField(max_length=100)
  832. installed_name = models.CharField(max_length=100, default='')
  833. version = models.CharField(max_length=100, blank=True)
  834. revision = models.CharField(max_length=32, blank=True)
  835. summary = models.TextField(blank=True)
  836. description = models.TextField(blank=True)
  837. size = models.IntegerField(default=0)
  838. installed_size = models.IntegerField(default=0)
  839. section = models.CharField(max_length=80, blank=True)
  840. license = models.CharField(max_length=80, blank=True)
  841. @property
  842. def is_locale_package(self):
  843. """ Returns True if this package is identifiable as a locale package """
  844. if self.name.find('locale') != -1:
  845. return True
  846. return False
  847. @property
  848. def is_packagegroup(self):
  849. """ Returns True is this package is identifiable as a packagegroup """
  850. if self.name.find('packagegroup') != -1:
  851. return True
  852. return False
  853. class CustomImagePackage(Package):
  854. # CustomImageRecipe fields to track pacakges appended,
  855. # included and excluded from a CustomImageRecipe
  856. recipe_includes = models.ManyToManyField('CustomImageRecipe',
  857. related_name='includes_set')
  858. recipe_excludes = models.ManyToManyField('CustomImageRecipe',
  859. related_name='excludes_set')
  860. recipe_appends = models.ManyToManyField('CustomImageRecipe',
  861. related_name='appends_set')
  862. class Package_DependencyManager(models.Manager):
  863. use_for_related_fields = True
  864. TARGET_LATEST = "use-latest-target-for-target"
  865. def get_queryset(self):
  866. return super(Package_DependencyManager, self).get_queryset().exclude(package_id = F('depends_on__id'))
  867. def for_target_or_none(self, target):
  868. """ filter the dependencies to be displayed by the supplied target
  869. if no dependences are found for the target then try None as the target
  870. which will return the dependences calculated without the context of a
  871. target e.g. non image recipes.
  872. returns: { size, packages }
  873. """
  874. package_dependencies = self.all_depends().order_by('depends_on__name')
  875. if target is self.TARGET_LATEST:
  876. installed_deps =\
  877. package_dependencies.filter(~Q(target__target=None))
  878. else:
  879. installed_deps =\
  880. package_dependencies.filter(Q(target__target=target))
  881. packages_list = None
  882. total_size = 0
  883. # If we have installed depdencies for this package and target then use
  884. # these to display
  885. if installed_deps.count() > 0:
  886. packages_list = installed_deps
  887. total_size = installed_deps.aggregate(
  888. Sum('depends_on__size'))['depends_on__size__sum']
  889. else:
  890. new_list = []
  891. package_names = []
  892. # Find dependencies for the package that we know about even if
  893. # it's not installed on a target e.g. from a non-image recipe
  894. for p in package_dependencies.filter(Q(target=None)):
  895. if p.depends_on.name in package_names:
  896. continue
  897. else:
  898. package_names.append(p.depends_on.name)
  899. new_list.append(p.pk)
  900. # while we're here we may as well total up the size to
  901. # avoid iterating again
  902. total_size += p.depends_on.size
  903. # We want to return a queryset here for consistency so pick the
  904. # deps from the new_list
  905. packages_list = package_dependencies.filter(Q(pk__in=new_list))
  906. return {'packages': packages_list,
  907. 'size': total_size}
  908. def all_depends(self):
  909. """ Returns just the depends packages and not any other dep_type
  910. Note that this is for any target
  911. """
  912. return self.filter(Q(dep_type=Package_Dependency.TYPE_RDEPENDS) |
  913. Q(dep_type=Package_Dependency.TYPE_TRDEPENDS))
  914. class Package_Dependency(models.Model):
  915. TYPE_RDEPENDS = 0
  916. TYPE_TRDEPENDS = 1
  917. TYPE_RRECOMMENDS = 2
  918. TYPE_TRECOMMENDS = 3
  919. TYPE_RSUGGESTS = 4
  920. TYPE_RPROVIDES = 5
  921. TYPE_RREPLACES = 6
  922. TYPE_RCONFLICTS = 7
  923. ' TODO: bpackage should be changed to remove the DEPENDS_TYPE access '
  924. DEPENDS_TYPE = (
  925. (TYPE_RDEPENDS, "depends"),
  926. (TYPE_TRDEPENDS, "depends"),
  927. (TYPE_TRECOMMENDS, "recommends"),
  928. (TYPE_RRECOMMENDS, "recommends"),
  929. (TYPE_RSUGGESTS, "suggests"),
  930. (TYPE_RPROVIDES, "provides"),
  931. (TYPE_RREPLACES, "replaces"),
  932. (TYPE_RCONFLICTS, "conflicts"),
  933. )
  934. """ Indexed by dep_type, in view order, key for short name and help
  935. description which when viewed will be printf'd with the
  936. package name.
  937. """
  938. DEPENDS_DICT = {
  939. TYPE_RDEPENDS : ("depends", "%s is required to run %s"),
  940. TYPE_TRDEPENDS : ("depends", "%s is required to run %s"),
  941. TYPE_TRECOMMENDS : ("recommends", "%s extends the usability of %s"),
  942. TYPE_RRECOMMENDS : ("recommends", "%s extends the usability of %s"),
  943. TYPE_RSUGGESTS : ("suggests", "%s is suggested for installation with %s"),
  944. TYPE_RPROVIDES : ("provides", "%s is provided by %s"),
  945. TYPE_RREPLACES : ("replaces", "%s is replaced by %s"),
  946. TYPE_RCONFLICTS : ("conflicts", "%s conflicts with %s, which will not be installed if this package is not first removed"),
  947. }
  948. package = models.ForeignKey(Package, related_name='package_dependencies_source')
  949. depends_on = models.ForeignKey(Package, related_name='package_dependencies_target') # soft dependency
  950. dep_type = models.IntegerField(choices=DEPENDS_TYPE)
  951. target = models.ForeignKey(Target, null=True)
  952. objects = Package_DependencyManager()
  953. class Target_Installed_Package(models.Model):
  954. target = models.ForeignKey(Target)
  955. package = models.ForeignKey(Package, related_name='buildtargetlist_package')
  956. class Package_File(models.Model):
  957. package = models.ForeignKey(Package, related_name='buildfilelist_package')
  958. path = models.FilePathField(max_length=255, blank=True)
  959. size = models.IntegerField()
  960. class Recipe(models.Model):
  961. search_allowed_fields = ['name', 'version', 'file_path', 'section',
  962. 'summary', 'description', 'license',
  963. 'layer_version__layer__name',
  964. 'layer_version__branch', 'layer_version__commit',
  965. 'layer_version__local_path',
  966. 'layer_version__layer_source']
  967. up_date = models.DateTimeField(null=True, default=None)
  968. name = models.CharField(max_length=100, blank=True)
  969. version = models.CharField(max_length=100, blank=True)
  970. layer_version = models.ForeignKey('Layer_Version',
  971. related_name='recipe_layer_version')
  972. summary = models.TextField(blank=True)
  973. description = models.TextField(blank=True)
  974. section = models.CharField(max_length=100, blank=True)
  975. license = models.CharField(max_length=200, blank=True)
  976. homepage = models.URLField(blank=True)
  977. bugtracker = models.URLField(blank=True)
  978. file_path = models.FilePathField(max_length=255)
  979. pathflags = models.CharField(max_length=200, blank=True)
  980. is_image = models.BooleanField(default=False)
  981. def __unicode__(self):
  982. return "Recipe " + self.name + ":" + self.version
  983. def get_vcs_recipe_file_link_url(self):
  984. return self.layer_version.get_vcs_file_link_url(self.file_path)
  985. def get_description_or_summary(self):
  986. if self.description:
  987. return self.description
  988. elif self.summary:
  989. return self.summary
  990. else:
  991. return ""
  992. class Meta:
  993. unique_together = (("layer_version", "file_path", "pathflags"), )
  994. class Recipe_DependencyManager(models.Manager):
  995. use_for_related_fields = True
  996. def get_queryset(self):
  997. return super(Recipe_DependencyManager, self).get_queryset().exclude(recipe_id = F('depends_on__id'))
  998. class Provides(models.Model):
  999. name = models.CharField(max_length=100)
  1000. recipe = models.ForeignKey(Recipe)
  1001. class Recipe_Dependency(models.Model):
  1002. TYPE_DEPENDS = 0
  1003. TYPE_RDEPENDS = 1
  1004. DEPENDS_TYPE = (
  1005. (TYPE_DEPENDS, "depends"),
  1006. (TYPE_RDEPENDS, "rdepends"),
  1007. )
  1008. recipe = models.ForeignKey(Recipe, related_name='r_dependencies_recipe')
  1009. depends_on = models.ForeignKey(Recipe, related_name='r_dependencies_depends')
  1010. via = models.ForeignKey(Provides, null=True, default=None)
  1011. dep_type = models.IntegerField(choices=DEPENDS_TYPE)
  1012. objects = Recipe_DependencyManager()
  1013. class Machine(models.Model):
  1014. search_allowed_fields = ["name", "description", "layer_version__layer__name"]
  1015. up_date = models.DateTimeField(null = True, default = None)
  1016. layer_version = models.ForeignKey('Layer_Version')
  1017. name = models.CharField(max_length=255)
  1018. description = models.CharField(max_length=255)
  1019. def get_vcs_machine_file_link_url(self):
  1020. path = 'conf/machine/'+self.name+'.conf'
  1021. return self.layer_version.get_vcs_file_link_url(path)
  1022. def __unicode__(self):
  1023. return "Machine " + self.name + "(" + self.description + ")"
  1024. class BitbakeVersion(models.Model):
  1025. name = models.CharField(max_length=32, unique = True)
  1026. giturl = GitURLField()
  1027. branch = models.CharField(max_length=32)
  1028. dirpath = models.CharField(max_length=255)
  1029. def __unicode__(self):
  1030. return "%s (Branch: %s)" % (self.name, self.branch)
  1031. class Release(models.Model):
  1032. """ A release is a project template, used to pre-populate Project settings with a configuration set """
  1033. name = models.CharField(max_length=32, unique = True)
  1034. description = models.CharField(max_length=255)
  1035. bitbake_version = models.ForeignKey(BitbakeVersion)
  1036. branch_name = models.CharField(max_length=50, default = "")
  1037. helptext = models.TextField(null=True)
  1038. def __unicode__(self):
  1039. return "%s (%s)" % (self.name, self.branch_name)
  1040. def __str__(self):
  1041. return self.name
  1042. class ReleaseDefaultLayer(models.Model):
  1043. release = models.ForeignKey(Release)
  1044. layer_name = models.CharField(max_length=100, default="")
  1045. class LayerSource(object):
  1046. """ Where the layer metadata came from """
  1047. TYPE_LOCAL = 0
  1048. TYPE_LAYERINDEX = 1
  1049. TYPE_IMPORTED = 2
  1050. TYPE_BUILD = 3
  1051. SOURCE_TYPE = (
  1052. (TYPE_LOCAL, "local"),
  1053. (TYPE_LAYERINDEX, "layerindex"),
  1054. (TYPE_IMPORTED, "imported"),
  1055. (TYPE_BUILD, "build"),
  1056. )
  1057. def types_dict():
  1058. """ Turn the TYPES enums into a simple dictionary """
  1059. dictionary = {}
  1060. for key in LayerSource.__dict__:
  1061. if "TYPE" in key:
  1062. dictionary[key] = getattr(LayerSource, key)
  1063. return dictionary
  1064. class Layer(models.Model):
  1065. up_date = models.DateTimeField(null=True, default=timezone.now)
  1066. name = models.CharField(max_length=100)
  1067. layer_index_url = models.URLField()
  1068. vcs_url = GitURLField(default=None, null=True)
  1069. vcs_web_url = models.URLField(null=True, default=None)
  1070. vcs_web_tree_base_url = models.URLField(null=True, default=None)
  1071. vcs_web_file_base_url = models.URLField(null=True, default=None)
  1072. summary = models.TextField(help_text='One-line description of the layer',
  1073. null=True, default=None)
  1074. description = models.TextField(null=True, default=None)
  1075. def __unicode__(self):
  1076. return "%s / %s " % (self.name, self.summary)
  1077. class Layer_Version(models.Model):
  1078. """
  1079. A Layer_Version either belongs to a single project or no project
  1080. """
  1081. search_allowed_fields = ["layer__name", "layer__summary",
  1082. "layer__description", "layer__vcs_url",
  1083. "dirpath", "release__name", "commit", "branch"]
  1084. build = models.ForeignKey(Build, related_name='layer_version_build',
  1085. default=None, null=True)
  1086. layer = models.ForeignKey(Layer, related_name='layer_version_layer')
  1087. layer_source = models.IntegerField(choices=LayerSource.SOURCE_TYPE,
  1088. default=0)
  1089. up_date = models.DateTimeField(null=True, default=timezone.now)
  1090. # To which metadata release does this layer version belong to
  1091. release = models.ForeignKey(Release, null=True, default=None)
  1092. branch = models.CharField(max_length=80)
  1093. commit = models.CharField(max_length=100)
  1094. # If the layer is in a subdir
  1095. dirpath = models.CharField(max_length=255, null=True, default=None)
  1096. # if -1, this is a default layer
  1097. priority = models.IntegerField(default=0)
  1098. # where this layer exists on the filesystem
  1099. local_path = models.FilePathField(max_length=1024, default="/")
  1100. # Set if this layer is restricted to a particular project
  1101. project = models.ForeignKey('Project', null=True, default=None)
  1102. # code lifted, with adaptations, from the layerindex-web application
  1103. # https://git.yoctoproject.org/cgit/cgit.cgi/layerindex-web/
  1104. def _handle_url_path(self, base_url, path):
  1105. import re, posixpath
  1106. if base_url:
  1107. if self.dirpath:
  1108. if path:
  1109. extra_path = self.dirpath + '/' + path
  1110. # Normalise out ../ in path for usage URL
  1111. extra_path = posixpath.normpath(extra_path)
  1112. # Minor workaround to handle case where subdirectory has been added between branches
  1113. # (should probably support usage URL per branch to handle this... sigh...)
  1114. if extra_path.startswith('../'):
  1115. extra_path = extra_path[3:]
  1116. else:
  1117. extra_path = self.dirpath
  1118. else:
  1119. extra_path = path
  1120. branchname = self.release.name
  1121. url = base_url.replace('%branch%', branchname)
  1122. # If there's a % in the path (e.g. a wildcard bbappend) we need to encode it
  1123. if extra_path:
  1124. extra_path = extra_path.replace('%', '%25')
  1125. if '%path%' in base_url:
  1126. if extra_path:
  1127. url = re.sub(r'\[([^\]]*%path%[^\]]*)\]', '\\1', url)
  1128. else:
  1129. url = re.sub(r'\[([^\]]*%path%[^\]]*)\]', '', url)
  1130. return url.replace('%path%', extra_path)
  1131. else:
  1132. return url + extra_path
  1133. return None
  1134. def get_vcs_link_url(self):
  1135. if self.layer.vcs_web_url is None:
  1136. return None
  1137. return self.layer.vcs_web_url
  1138. def get_vcs_file_link_url(self, file_path=""):
  1139. if self.layer.vcs_web_file_base_url is None:
  1140. return None
  1141. return self._handle_url_path(self.layer.vcs_web_file_base_url,
  1142. file_path)
  1143. def get_vcs_dirpath_link_url(self):
  1144. if self.layer.vcs_web_tree_base_url is None:
  1145. return None
  1146. return self._handle_url_path(self.layer.vcs_web_tree_base_url, '')
  1147. def get_vcs_reference(self):
  1148. if self.branch is not None and len(self.branch) > 0:
  1149. return self.branch
  1150. if self.release is not None:
  1151. return self.release.name
  1152. if self.commit is not None and len(self.commit) > 0:
  1153. return self.commit
  1154. return 'N/A'
  1155. def get_detailspage_url(self, project_id):
  1156. return reverse('layerdetails', args=(project_id, self.pk))
  1157. def get_alldeps(self, project_id):
  1158. """Get full list of unique layer dependencies."""
  1159. def gen_layerdeps(lver, project):
  1160. for ldep in lver.dependencies.all():
  1161. yield ldep.depends_on
  1162. # get next level of deps recursively calling gen_layerdeps
  1163. for subdep in gen_layerdeps(ldep.depends_on, project):
  1164. yield subdep
  1165. project = Project.objects.get(pk=project_id)
  1166. result = []
  1167. projectlvers = [player.layercommit for player in project.projectlayer_set.all()]
  1168. for dep in gen_layerdeps(self, project):
  1169. # filter out duplicates and layers already belonging to the project
  1170. if dep not in result + projectlvers:
  1171. result.append(dep)
  1172. return sorted(result, key=lambda x: x.layer.name)
  1173. def __unicode__(self):
  1174. return ("id %d belongs to layer: %s" % (self.pk, self.layer.name))
  1175. def __str__(self):
  1176. if self.release:
  1177. release = self.release.name
  1178. else:
  1179. release = "No release set"
  1180. return "%d %s (%s)" % (self.pk, self.layer.name, release)
  1181. class LayerVersionDependency(models.Model):
  1182. layer_version = models.ForeignKey(Layer_Version,
  1183. related_name="dependencies")
  1184. depends_on = models.ForeignKey(Layer_Version,
  1185. related_name="dependees")
  1186. class ProjectLayer(models.Model):
  1187. project = models.ForeignKey(Project)
  1188. layercommit = models.ForeignKey(Layer_Version, null=True)
  1189. optional = models.BooleanField(default = True)
  1190. def __unicode__(self):
  1191. return "%s, %s" % (self.project.name, self.layercommit)
  1192. class Meta:
  1193. unique_together = (("project", "layercommit"),)
  1194. class CustomImageRecipe(Recipe):
  1195. # CustomImageRecipe's belong to layers called:
  1196. LAYER_NAME = "toaster-custom-images"
  1197. search_allowed_fields = ['name']
  1198. base_recipe = models.ForeignKey(Recipe, related_name='based_on_recipe')
  1199. project = models.ForeignKey(Project)
  1200. last_updated = models.DateTimeField(null=True, default=None)
  1201. def get_last_successful_built_target(self):
  1202. """ Return the last successful built target object if one exists
  1203. otherwise return None """
  1204. return Target.objects.filter(Q(build__outcome=Build.SUCCEEDED) &
  1205. Q(build__project=self.project) &
  1206. Q(target=self.name)).last()
  1207. def update_package_list(self):
  1208. """ Update the package list from the last good build of this
  1209. CustomImageRecipe
  1210. """
  1211. # Check if we're aldready up-to-date or not
  1212. target = self.get_last_successful_built_target()
  1213. if target == None:
  1214. # So we've never actually built this Custom recipe but what about
  1215. # the recipe it's based on?
  1216. target = \
  1217. Target.objects.filter(Q(build__outcome=Build.SUCCEEDED) &
  1218. Q(build__project=self.project) &
  1219. Q(target=self.base_recipe.name)).last()
  1220. if target == None:
  1221. return
  1222. if target.build.completed_on == self.last_updated:
  1223. return
  1224. self.includes_set.clear()
  1225. excludes_list = self.excludes_set.values_list('name', flat=True)
  1226. appends_list = self.appends_set.values_list('name', flat=True)
  1227. built_packages_list = \
  1228. target.target_installed_package_set.values_list('package__name',
  1229. flat=True)
  1230. for built_package in built_packages_list:
  1231. # Is the built package in the custom packages list?
  1232. if built_package in excludes_list:
  1233. continue
  1234. if built_package in appends_list:
  1235. continue
  1236. cust_img_p = \
  1237. CustomImagePackage.objects.get(name=built_package)
  1238. self.includes_set.add(cust_img_p)
  1239. self.last_updated = target.build.completed_on
  1240. self.save()
  1241. def get_all_packages(self):
  1242. """Get the included packages and any appended packages"""
  1243. self.update_package_list()
  1244. return CustomImagePackage.objects.filter((Q(recipe_appends=self) |
  1245. Q(recipe_includes=self)) &
  1246. ~Q(recipe_excludes=self))
  1247. def get_base_recipe_file(self):
  1248. """Get the base recipe file path if it exists on the file system"""
  1249. path_schema_one = "%s/%s" % (self.base_recipe.layer_version.dirpath,
  1250. self.base_recipe.file_path)
  1251. path_schema_two = self.base_recipe.file_path
  1252. if os.path.exists(path_schema_one):
  1253. return path_schema_one
  1254. # The path may now be the full path if the recipe has been built
  1255. if os.path.exists(path_schema_two):
  1256. return path_schema_two
  1257. return None
  1258. def generate_recipe_file_contents(self):
  1259. """Generate the contents for the recipe file."""
  1260. # If we have no excluded packages we only need to _append
  1261. if self.excludes_set.count() == 0:
  1262. packages_conf = "IMAGE_INSTALL_append = \" "
  1263. for pkg in self.appends_set.all():
  1264. packages_conf += pkg.name+' '
  1265. else:
  1266. packages_conf = "IMAGE_FEATURES =\"\"\nIMAGE_INSTALL = \""
  1267. # We add all the known packages to be built by this recipe apart
  1268. # from locale packages which are are controlled with IMAGE_LINGUAS.
  1269. for pkg in self.get_all_packages().exclude(
  1270. name__icontains="locale"):
  1271. packages_conf += pkg.name+' '
  1272. packages_conf += "\""
  1273. base_recipe_path = self.get_base_recipe_file()
  1274. if base_recipe_path:
  1275. base_recipe = open(base_recipe_path, 'r').read()
  1276. else:
  1277. raise IOError("Based on recipe file not found")
  1278. # Add a special case for when the recipe we have based a custom image
  1279. # recipe on requires another recipe.
  1280. # For example:
  1281. # "require core-image-minimal.bb" is changed to:
  1282. # "require recipes-core/images/core-image-minimal.bb"
  1283. req_search = re.search(r'(require\s+)(.+\.bb\s*$)',
  1284. base_recipe,
  1285. re.MULTILINE)
  1286. if req_search:
  1287. require_filename = req_search.group(2).strip()
  1288. corrected_location = Recipe.objects.filter(
  1289. Q(layer_version=self.base_recipe.layer_version) &
  1290. Q(file_path__icontains=require_filename)).last().file_path
  1291. new_require_line = "require %s" % corrected_location
  1292. base_recipe = base_recipe.replace(req_search.group(0),
  1293. new_require_line)
  1294. info = {
  1295. "date": timezone.now().strftime("%Y-%m-%d %H:%M:%S"),
  1296. "base_recipe": base_recipe,
  1297. "recipe_name": self.name,
  1298. "base_recipe_name": self.base_recipe.name,
  1299. "license": self.license,
  1300. "summary": self.summary,
  1301. "description": self.description,
  1302. "packages_conf": packages_conf.strip()
  1303. }
  1304. recipe_contents = ("# Original recipe %(base_recipe_name)s \n"
  1305. "%(base_recipe)s\n\n"
  1306. "# Recipe %(recipe_name)s \n"
  1307. "# Customisation Generated by Toaster on %(date)s\n"
  1308. "SUMMARY = \"%(summary)s\"\n"
  1309. "DESCRIPTION = \"%(description)s\"\n"
  1310. "LICENSE = \"%(license)s\"\n"
  1311. "%(packages_conf)s") % info
  1312. return recipe_contents
  1313. class ProjectVariable(models.Model):
  1314. project = models.ForeignKey(Project)
  1315. name = models.CharField(max_length=100)
  1316. value = models.TextField(blank = True)
  1317. class Variable(models.Model):
  1318. search_allowed_fields = ['variable_name', 'variable_value',
  1319. 'vhistory__file_name', "description"]
  1320. build = models.ForeignKey(Build, related_name='variable_build')
  1321. variable_name = models.CharField(max_length=100)
  1322. variable_value = models.TextField(blank=True)
  1323. changed = models.BooleanField(default=False)
  1324. human_readable_name = models.CharField(max_length=200)
  1325. description = models.TextField(blank=True)
  1326. class VariableHistory(models.Model):
  1327. variable = models.ForeignKey(Variable, related_name='vhistory')
  1328. value = models.TextField(blank=True)
  1329. file_name = models.FilePathField(max_length=255)
  1330. line_number = models.IntegerField(null=True)
  1331. operation = models.CharField(max_length=64)
  1332. class HelpText(models.Model):
  1333. VARIABLE = 0
  1334. HELPTEXT_AREA = ((VARIABLE, 'variable'), )
  1335. build = models.ForeignKey(Build, related_name='helptext_build')
  1336. area = models.IntegerField(choices=HELPTEXT_AREA)
  1337. key = models.CharField(max_length=100)
  1338. text = models.TextField()
  1339. class LogMessage(models.Model):
  1340. EXCEPTION = -1 # used to signal self-toaster-exceptions
  1341. INFO = 0
  1342. WARNING = 1
  1343. ERROR = 2
  1344. CRITICAL = 3
  1345. LOG_LEVEL = (
  1346. (INFO, "info"),
  1347. (WARNING, "warn"),
  1348. (ERROR, "error"),
  1349. (CRITICAL, "critical"),
  1350. (EXCEPTION, "toaster exception")
  1351. )
  1352. build = models.ForeignKey(Build)
  1353. task = models.ForeignKey(Task, blank = True, null=True)
  1354. level = models.IntegerField(choices=LOG_LEVEL, default=INFO)
  1355. message = models.TextField(blank=True, null=True)
  1356. pathname = models.FilePathField(max_length=255, blank=True)
  1357. lineno = models.IntegerField(null=True)
  1358. def __str__(self):
  1359. return force_bytes('%s %s %s' % (self.get_level_display(), self.message, self.build))
  1360. def invalidate_cache(**kwargs):
  1361. from django.core.cache import cache
  1362. try:
  1363. cache.clear()
  1364. except Exception as e:
  1365. logger.warning("Problem with cache backend: Failed to clear cache: %s" % e)
  1366. django.db.models.signals.post_save.connect(invalidate_cache)
  1367. django.db.models.signals.post_delete.connect(invalidate_cache)
  1368. django.db.models.signals.m2m_changed.connect(invalidate_cache)