knotty.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599
  1. #
  2. # BitBake (No)TTY UI Implementation
  3. #
  4. # Handling output to TTYs or files (no TTY)
  5. #
  6. # Copyright (C) 2006-2012 Richard Purdie
  7. #
  8. # This program is free software; you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License version 2 as
  10. # published by the Free Software Foundation.
  11. #
  12. # This program is distributed in the hope that it will be useful,
  13. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. # GNU General Public License for more details.
  16. #
  17. # You should have received a copy of the GNU General Public License along
  18. # with this program; if not, write to the Free Software Foundation, Inc.,
  19. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  20. from __future__ import division
  21. import os
  22. import sys
  23. import xmlrpclib
  24. import logging
  25. import progressbar
  26. import signal
  27. import bb.msg
  28. import time
  29. import fcntl
  30. import struct
  31. import copy
  32. import atexit
  33. from bb.ui import uihelper
  34. featureSet = [bb.cooker.CookerFeatures.SEND_SANITYEVENTS]
  35. logger = logging.getLogger("BitBake")
  36. interactive = sys.stdout.isatty()
  37. class BBProgress(progressbar.ProgressBar):
  38. def __init__(self, msg, maxval):
  39. self.msg = msg
  40. widgets = [progressbar.Percentage(), ' ', progressbar.Bar(), ' ',
  41. progressbar.ETA()]
  42. try:
  43. self._resize_default = signal.getsignal(signal.SIGWINCH)
  44. except:
  45. self._resize_default = None
  46. progressbar.ProgressBar.__init__(self, maxval, [self.msg + ": "] + widgets, fd=sys.stdout)
  47. def _handle_resize(self, signum, frame):
  48. progressbar.ProgressBar._handle_resize(self, signum, frame)
  49. if self._resize_default:
  50. self._resize_default(signum, frame)
  51. def finish(self):
  52. progressbar.ProgressBar.finish(self)
  53. if self._resize_default:
  54. signal.signal(signal.SIGWINCH, self._resize_default)
  55. class NonInteractiveProgress(object):
  56. fobj = sys.stdout
  57. def __init__(self, msg, maxval):
  58. self.msg = msg
  59. self.maxval = maxval
  60. def start(self):
  61. self.fobj.write("%s..." % self.msg)
  62. self.fobj.flush()
  63. return self
  64. def update(self, value):
  65. pass
  66. def finish(self):
  67. self.fobj.write("done.\n")
  68. self.fobj.flush()
  69. def new_progress(msg, maxval):
  70. if interactive:
  71. return BBProgress(msg, maxval)
  72. else:
  73. return NonInteractiveProgress(msg, maxval)
  74. def pluralise(singular, plural, qty):
  75. if(qty == 1):
  76. return singular % qty
  77. else:
  78. return plural % qty
  79. class InteractConsoleLogFilter(logging.Filter):
  80. def __init__(self, tf, format):
  81. self.tf = tf
  82. self.format = format
  83. def filter(self, record):
  84. if record.levelno == self.format.NOTE and (record.msg.startswith("Running") or record.msg.startswith("recipe ")):
  85. return False
  86. self.tf.clearFooter()
  87. return True
  88. class TerminalFilter(object):
  89. rows = 25
  90. columns = 80
  91. def sigwinch_handle(self, signum, frame):
  92. self.rows, self.columns = self.getTerminalColumns()
  93. if self._sigwinch_default:
  94. self._sigwinch_default(signum, frame)
  95. def getTerminalColumns(self):
  96. def ioctl_GWINSZ(fd):
  97. try:
  98. cr = struct.unpack('hh', fcntl.ioctl(fd, self.termios.TIOCGWINSZ, '1234'))
  99. except:
  100. return None
  101. return cr
  102. cr = ioctl_GWINSZ(sys.stdout.fileno())
  103. if not cr:
  104. try:
  105. fd = os.open(os.ctermid(), os.O_RDONLY)
  106. cr = ioctl_GWINSZ(fd)
  107. os.close(fd)
  108. except:
  109. pass
  110. if not cr:
  111. try:
  112. cr = (env['LINES'], env['COLUMNS'])
  113. except:
  114. cr = (25, 80)
  115. return cr
  116. def __init__(self, main, helper, console, errconsole, format):
  117. self.main = main
  118. self.helper = helper
  119. self.cuu = None
  120. self.stdinbackup = None
  121. self.interactive = sys.stdout.isatty()
  122. self.footer_present = False
  123. self.lastpids = []
  124. if not self.interactive:
  125. return
  126. try:
  127. import curses
  128. except ImportError:
  129. sys.exit("FATAL: The knotty ui could not load the required curses python module.")
  130. import termios
  131. self.curses = curses
  132. self.termios = termios
  133. try:
  134. fd = sys.stdin.fileno()
  135. self.stdinbackup = termios.tcgetattr(fd)
  136. new = copy.deepcopy(self.stdinbackup)
  137. new[3] = new[3] & ~termios.ECHO
  138. termios.tcsetattr(fd, termios.TCSADRAIN, new)
  139. curses.setupterm()
  140. if curses.tigetnum("colors") > 2:
  141. format.enable_color()
  142. self.ed = curses.tigetstr("ed")
  143. if self.ed:
  144. self.cuu = curses.tigetstr("cuu")
  145. try:
  146. self._sigwinch_default = signal.getsignal(signal.SIGWINCH)
  147. signal.signal(signal.SIGWINCH, self.sigwinch_handle)
  148. except:
  149. pass
  150. self.rows, self.columns = self.getTerminalColumns()
  151. except:
  152. self.cuu = None
  153. if not self.cuu:
  154. self.interactive = False
  155. bb.note("Unable to use interactive mode for this terminal, using fallback")
  156. return
  157. console.addFilter(InteractConsoleLogFilter(self, format))
  158. errconsole.addFilter(InteractConsoleLogFilter(self, format))
  159. def clearFooter(self):
  160. if self.footer_present:
  161. lines = self.footer_present
  162. sys.stdout.write(self.curses.tparm(self.cuu, lines))
  163. sys.stdout.write(self.curses.tparm(self.ed))
  164. self.footer_present = False
  165. def updateFooter(self):
  166. if not self.cuu:
  167. return
  168. activetasks = self.helper.running_tasks
  169. failedtasks = self.helper.failed_tasks
  170. runningpids = self.helper.running_pids
  171. if self.footer_present and (self.lastcount == self.helper.tasknumber_current) and (self.lastpids == runningpids):
  172. return
  173. if self.footer_present:
  174. self.clearFooter()
  175. if (not self.helper.tasknumber_total or self.helper.tasknumber_current == self.helper.tasknumber_total) and not len(activetasks):
  176. return
  177. tasks = []
  178. for t in runningpids:
  179. tasks.append("%s (pid %s)" % (activetasks[t]["title"], t))
  180. if self.main.shutdown:
  181. content = "Waiting for %s running tasks to finish:" % len(activetasks)
  182. elif not len(activetasks):
  183. content = "No currently running tasks (%s of %s)" % (self.helper.tasknumber_current, self.helper.tasknumber_total)
  184. else:
  185. content = "Currently %s running tasks (%s of %s):" % (len(activetasks), self.helper.tasknumber_current, self.helper.tasknumber_total)
  186. print(content)
  187. lines = 1 + int(len(content) / (self.columns + 1))
  188. for tasknum, task in enumerate(tasks[:(self.rows - 2)]):
  189. content = "%s: %s" % (tasknum, task)
  190. print(content)
  191. lines = lines + 1 + int(len(content) / (self.columns + 1))
  192. self.footer_present = lines
  193. self.lastpids = runningpids[:]
  194. self.lastcount = self.helper.tasknumber_current
  195. def finish(self):
  196. if self.stdinbackup:
  197. fd = sys.stdin.fileno()
  198. self.termios.tcsetattr(fd, self.termios.TCSADRAIN, self.stdinbackup)
  199. def _log_settings_from_server(server):
  200. # Get values of variables which control our output
  201. includelogs, error = server.runCommand(["getVariable", "BBINCLUDELOGS"])
  202. if error:
  203. logger.error("Unable to get the value of BBINCLUDELOGS variable: %s" % error)
  204. raise BaseException(error)
  205. loglines, error = server.runCommand(["getVariable", "BBINCLUDELOGS_LINES"])
  206. if error:
  207. logger.error("Unable to get the value of BBINCLUDELOGS_LINES variable: %s" % error)
  208. raise BaseException(error)
  209. consolelogfile, error = server.runCommand(["getSetVariable", "BB_CONSOLELOG"])
  210. if error:
  211. logger.error("Unable to get the value of BB_CONSOLELOG variable: %s" % error)
  212. raise BaseException(error)
  213. return includelogs, loglines, consolelogfile
  214. _evt_list = [ "bb.runqueue.runQueueExitWait", "bb.event.LogExecTTY", "logging.LogRecord",
  215. "bb.build.TaskFailed", "bb.build.TaskBase", "bb.event.ParseStarted",
  216. "bb.event.ParseProgress", "bb.event.ParseCompleted", "bb.event.CacheLoadStarted",
  217. "bb.event.CacheLoadProgress", "bb.event.CacheLoadCompleted", "bb.command.CommandFailed",
  218. "bb.command.CommandExit", "bb.command.CommandCompleted", "bb.cooker.CookerExit",
  219. "bb.event.MultipleProviders", "bb.event.NoProvider", "bb.runqueue.sceneQueueTaskStarted",
  220. "bb.runqueue.runQueueTaskStarted", "bb.runqueue.runQueueTaskFailed", "bb.runqueue.sceneQueueTaskFailed",
  221. "bb.event.BuildBase", "bb.build.TaskStarted", "bb.build.TaskSucceeded", "bb.build.TaskFailedSilent"]
  222. def main(server, eventHandler, params, tf = TerminalFilter):
  223. includelogs, loglines, consolelogfile = _log_settings_from_server(server)
  224. if sys.stdin.isatty() and sys.stdout.isatty():
  225. log_exec_tty = True
  226. else:
  227. log_exec_tty = False
  228. helper = uihelper.BBUIHelper()
  229. console = logging.StreamHandler(sys.stdout)
  230. errconsole = logging.StreamHandler(sys.stderr)
  231. format_str = "%(levelname)s: %(message)s"
  232. format = bb.msg.BBLogFormatter(format_str)
  233. bb.msg.addDefaultlogFilter(console, bb.msg.BBLogFilterStdOut)
  234. bb.msg.addDefaultlogFilter(errconsole, bb.msg.BBLogFilterStdErr)
  235. console.setFormatter(format)
  236. errconsole.setFormatter(format)
  237. logger.addHandler(console)
  238. logger.addHandler(errconsole)
  239. bb.utils.set_process_name("KnottyUI")
  240. if params.options.remote_server and params.options.kill_server:
  241. server.terminateServer()
  242. return
  243. consolelog = None
  244. if consolelogfile and not params.options.show_environment and not params.options.show_versions:
  245. bb.utils.mkdirhier(os.path.dirname(consolelogfile))
  246. conlogformat = bb.msg.BBLogFormatter(format_str)
  247. consolelog = logging.FileHandler(consolelogfile)
  248. bb.msg.addDefaultlogFilter(consolelog)
  249. consolelog.setFormatter(conlogformat)
  250. logger.addHandler(consolelog)
  251. llevel, debug_domains = bb.msg.constructLogOptions()
  252. server.runCommand(["setEventMask", server.getEventHandle(), llevel, debug_domains, _evt_list])
  253. universe = False
  254. if not params.observe_only:
  255. params.updateFromServer(server)
  256. params.updateToServer(server, os.environ.copy())
  257. cmdline = params.parseActions()
  258. if not cmdline:
  259. print("Nothing to do. Use 'bitbake world' to build everything, or run 'bitbake --help' for usage information.")
  260. return 1
  261. if 'msg' in cmdline and cmdline['msg']:
  262. logger.error(cmdline['msg'])
  263. return 1
  264. if cmdline['action'][0] == "buildTargets" and "universe" in cmdline['action'][1]:
  265. universe = True
  266. ret, error = server.runCommand(cmdline['action'])
  267. if error:
  268. logger.error("Command '%s' failed: %s" % (cmdline, error))
  269. return 1
  270. elif ret != True:
  271. logger.error("Command '%s' failed: returned %s" % (cmdline, ret))
  272. return 1
  273. parseprogress = None
  274. cacheprogress = None
  275. main.shutdown = 0
  276. interrupted = False
  277. return_value = 0
  278. errors = 0
  279. warnings = 0
  280. taskfailures = []
  281. termfilter = tf(main, helper, console, errconsole, format)
  282. atexit.register(termfilter.finish)
  283. while True:
  284. try:
  285. event = eventHandler.waitEvent(0)
  286. if event is None:
  287. if main.shutdown > 1:
  288. break
  289. termfilter.updateFooter()
  290. event = eventHandler.waitEvent(0.25)
  291. if event is None:
  292. continue
  293. helper.eventHandler(event)
  294. if isinstance(event, bb.runqueue.runQueueExitWait):
  295. if not main.shutdown:
  296. main.shutdown = 1
  297. continue
  298. if isinstance(event, bb.event.LogExecTTY):
  299. if log_exec_tty:
  300. tries = event.retries
  301. while tries:
  302. print("Trying to run: %s" % event.prog)
  303. if os.system(event.prog) == 0:
  304. break
  305. time.sleep(event.sleep_delay)
  306. tries -= 1
  307. if tries:
  308. continue
  309. logger.warning(event.msg)
  310. continue
  311. if isinstance(event, logging.LogRecord):
  312. if event.levelno >= format.ERROR:
  313. errors = errors + 1
  314. return_value = 1
  315. elif event.levelno == format.WARNING:
  316. warnings = warnings + 1
  317. if event.taskpid != 0:
  318. # For "normal" logging conditions, don't show note logs from tasks
  319. # but do show them if the user has changed the default log level to
  320. # include verbose/debug messages
  321. if event.levelno <= format.NOTE and (event.levelno < llevel or (event.levelno == format.NOTE and llevel != format.VERBOSE)):
  322. continue
  323. # Prefix task messages with recipe/task
  324. if event.taskpid in helper.running_tasks:
  325. taskinfo = helper.running_tasks[event.taskpid]
  326. event.msg = taskinfo['title'] + ': ' + event.msg
  327. if hasattr(event, 'fn'):
  328. event.msg = event.fn + ': ' + event.msg
  329. logger.handle(event)
  330. continue
  331. if isinstance(event, bb.build.TaskFailedSilent):
  332. logger.warning("Logfile for failed setscene task is %s" % event.logfile)
  333. continue
  334. if isinstance(event, bb.build.TaskFailed):
  335. return_value = 1
  336. logfile = event.logfile
  337. if logfile and os.path.exists(logfile):
  338. termfilter.clearFooter()
  339. bb.error("Logfile of failure stored in: %s" % logfile)
  340. if includelogs and not event.errprinted:
  341. print("Log data follows:")
  342. f = open(logfile, "r")
  343. lines = []
  344. while True:
  345. l = f.readline()
  346. if l == '':
  347. break
  348. l = l.rstrip()
  349. if loglines:
  350. lines.append(' | %s' % l)
  351. if len(lines) > int(loglines):
  352. lines.pop(0)
  353. else:
  354. print('| %s' % l)
  355. f.close()
  356. if lines:
  357. for line in lines:
  358. print(line)
  359. if isinstance(event, bb.build.TaskBase):
  360. logger.info(event._message)
  361. continue
  362. if isinstance(event, bb.event.ParseStarted):
  363. if event.total == 0:
  364. continue
  365. parseprogress = new_progress("Parsing recipes", event.total).start()
  366. continue
  367. if isinstance(event, bb.event.ParseProgress):
  368. parseprogress.update(event.current)
  369. continue
  370. if isinstance(event, bb.event.ParseCompleted):
  371. if not parseprogress:
  372. continue
  373. parseprogress.finish()
  374. print(("Parsing of %d .bb files complete (%d cached, %d parsed). %d targets, %d skipped, %d masked, %d errors."
  375. % ( event.total, event.cached, event.parsed, event.virtuals, event.skipped, event.masked, event.errors)))
  376. continue
  377. if isinstance(event, bb.event.CacheLoadStarted):
  378. cacheprogress = new_progress("Loading cache", event.total).start()
  379. continue
  380. if isinstance(event, bb.event.CacheLoadProgress):
  381. cacheprogress.update(event.current)
  382. continue
  383. if isinstance(event, bb.event.CacheLoadCompleted):
  384. cacheprogress.finish()
  385. print("Loaded %d entries from dependency cache." % event.num_entries)
  386. continue
  387. if isinstance(event, bb.command.CommandFailed):
  388. return_value = event.exitcode
  389. if event.error:
  390. errors = errors + 1
  391. logger.error("Command execution failed: %s", event.error)
  392. main.shutdown = 2
  393. continue
  394. if isinstance(event, bb.command.CommandExit):
  395. if not return_value:
  396. return_value = event.exitcode
  397. continue
  398. if isinstance(event, (bb.command.CommandCompleted, bb.cooker.CookerExit)):
  399. main.shutdown = 2
  400. continue
  401. if isinstance(event, bb.event.MultipleProviders):
  402. logger.info("multiple providers are available for %s%s (%s)", event._is_runtime and "runtime " or "",
  403. event._item,
  404. ", ".join(event._candidates))
  405. rtime = ""
  406. if event._is_runtime:
  407. rtime = "R"
  408. logger.info("consider defining a PREFERRED_%sPROVIDER entry to match %s" % (rtime, event._item))
  409. continue
  410. if isinstance(event, bb.event.NoProvider):
  411. if event._runtime:
  412. r = "R"
  413. else:
  414. r = ""
  415. extra = ''
  416. if not event._reasons:
  417. if event._close_matches:
  418. extra = ". Close matches:\n %s" % '\n '.join(event._close_matches)
  419. # For universe builds, only show these as warnings, not errors
  420. h = logger.warning
  421. if not universe:
  422. return_value = 1
  423. errors = errors + 1
  424. h = logger.error
  425. if event._dependees:
  426. h("Nothing %sPROVIDES '%s' (but %s %sDEPENDS on or otherwise requires it)%s", r, event._item, ", ".join(event._dependees), r, extra)
  427. else:
  428. h("Nothing %sPROVIDES '%s'%s", r, event._item, extra)
  429. if event._reasons:
  430. for reason in event._reasons:
  431. h("%s", reason)
  432. continue
  433. if isinstance(event, bb.runqueue.sceneQueueTaskStarted):
  434. logger.info("Running setscene task %d of %d (%s)" % (event.stats.completed + event.stats.active + event.stats.failed + 1, event.stats.total, event.taskstring))
  435. continue
  436. if isinstance(event, bb.runqueue.runQueueTaskStarted):
  437. if event.noexec:
  438. tasktype = 'noexec task'
  439. else:
  440. tasktype = 'task'
  441. logger.info("Running %s %s of %s (ID: %s, %s)",
  442. tasktype,
  443. event.stats.completed + event.stats.active +
  444. event.stats.failed + 1,
  445. event.stats.total, event.taskid, event.taskstring)
  446. continue
  447. if isinstance(event, bb.runqueue.runQueueTaskFailed):
  448. return_value = 1
  449. taskfailures.append(event.taskstring)
  450. logger.error("Task %s (%s) failed with exit code '%s'",
  451. event.taskid, event.taskstring, event.exitcode)
  452. continue
  453. if isinstance(event, bb.runqueue.sceneQueueTaskFailed):
  454. logger.warning("Setscene task %s (%s) failed with exit code '%s' - real task will be run instead",
  455. event.taskid, event.taskstring, event.exitcode)
  456. continue
  457. if isinstance(event, bb.event.DepTreeGenerated):
  458. continue
  459. # ignore
  460. if isinstance(event, (bb.event.BuildBase,
  461. bb.event.MetadataEvent,
  462. bb.event.StampUpdate,
  463. bb.event.ConfigParsed,
  464. bb.event.RecipeParsed,
  465. bb.event.RecipePreFinalise,
  466. bb.runqueue.runQueueEvent,
  467. bb.event.OperationStarted,
  468. bb.event.OperationCompleted,
  469. bb.event.OperationProgress,
  470. bb.event.DiskFull)):
  471. continue
  472. logger.error("Unknown event: %s", event)
  473. except EnvironmentError as ioerror:
  474. termfilter.clearFooter()
  475. # ignore interrupted io
  476. if ioerror.args[0] == 4:
  477. continue
  478. sys.stderr.write(str(ioerror))
  479. if not params.observe_only:
  480. _, error = server.runCommand(["stateForceShutdown"])
  481. main.shutdown = 2
  482. except KeyboardInterrupt:
  483. termfilter.clearFooter()
  484. if params.observe_only:
  485. print("\nKeyboard Interrupt, exiting observer...")
  486. main.shutdown = 2
  487. if not params.observe_only and main.shutdown == 1:
  488. print("\nSecond Keyboard Interrupt, stopping...\n")
  489. _, error = server.runCommand(["stateForceShutdown"])
  490. if error:
  491. logger.error("Unable to cleanly stop: %s" % error)
  492. if not params.observe_only and main.shutdown == 0:
  493. print("\nKeyboard Interrupt, closing down...\n")
  494. interrupted = True
  495. _, error = server.runCommand(["stateShutdown"])
  496. if error:
  497. logger.error("Unable to cleanly shutdown: %s" % error)
  498. main.shutdown = main.shutdown + 1
  499. pass
  500. except Exception as e:
  501. import traceback
  502. sys.stderr.write(traceback.format_exc())
  503. if not params.observe_only:
  504. _, error = server.runCommand(["stateForceShutdown"])
  505. main.shutdown = 2
  506. return_value = 1
  507. try:
  508. summary = ""
  509. if taskfailures:
  510. summary += pluralise("\nSummary: %s task failed:",
  511. "\nSummary: %s tasks failed:", len(taskfailures))
  512. for failure in taskfailures:
  513. summary += "\n %s" % failure
  514. if warnings:
  515. summary += pluralise("\nSummary: There was %s WARNING message shown.",
  516. "\nSummary: There were %s WARNING messages shown.", warnings)
  517. if return_value and errors:
  518. summary += pluralise("\nSummary: There was %s ERROR message shown, returning a non-zero exit code.",
  519. "\nSummary: There were %s ERROR messages shown, returning a non-zero exit code.", errors)
  520. if summary:
  521. print(summary)
  522. if interrupted:
  523. print("Execution was interrupted, returning a non-zero exit code.")
  524. if return_value == 0:
  525. return_value = 1
  526. except IOError as e:
  527. import errno
  528. if e.errno == errno.EPIPE:
  529. pass
  530. if consolelog:
  531. logger.removeHandler(consolelog)
  532. consolelog.close()
  533. return return_value