ls.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. """List packages according to various criteria
  2. @copyright: 2014, Ansgar Burchardt <ansgar@debian.org>
  3. @license: GPL-2+
  4. """
  5. # This program is free software; you can redistribute it and/or modify
  6. # it under the terms of the GNU General Public License as published by
  7. # the Free Software Foundation; either version 2 of the License, or
  8. # (at your option) any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program; if not, write to the Free Software
  17. # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  18. from collections import defaultdict
  19. import sqlalchemy.dialects.postgresql as pgsql
  20. import sqlalchemy.sql as sql
  21. from daklib.dbconn import DBConn
  22. def list_packages(
  23. packages,
  24. suites=None,
  25. components=None,
  26. architectures=None,
  27. binary_types=None,
  28. source_and_binary=False,
  29. regex=False,
  30. format=None,
  31. highest=None,
  32. ):
  33. session = DBConn().session()
  34. try:
  35. t = DBConn().view_package_list
  36. comparison_operator = "~" if regex else "="
  37. where = sql.false()
  38. for package in packages:
  39. where = where | t.c.package.op(comparison_operator)(package)
  40. if source_and_binary:
  41. where = where | t.c.source.op(comparison_operator)(package)
  42. if suites is not None:
  43. where = where & (t.c.suite.in_(suites) | t.c.codename.in_(suites))
  44. if components is not None:
  45. where = where & t.c.component.in_(components)
  46. if architectures is not None:
  47. where = where & t.c.architecture.in_(architectures)
  48. if binary_types is not None:
  49. where = where & t.c.type.in_(binary_types)
  50. if format is None:
  51. c_architectures = sql.func.string_agg(
  52. t.c.architecture,
  53. pgsql.aggregate_order_by(
  54. ", ", t.c.architecture_is_source.desc(), t.c.architecture
  55. ),
  56. )
  57. query = (
  58. sql.select(
  59. [t.c.package, t.c.version, t.c.display_suite, c_architectures]
  60. )
  61. .where(where)
  62. .group_by(t.c.package, t.c.version, t.c.display_suite)
  63. .order_by(t.c.package, t.c.version, t.c.display_suite)
  64. )
  65. result = session.execute(query).fetchall()
  66. if len(result) == 0:
  67. return
  68. lengths = {
  69. "package": max(10, max(len(row[t.c.package]) for row in result)),
  70. "version": max(13, max(len(row[t.c.version]) for row in result)),
  71. "suite": max(10, max(len(row[t.c.display_suite]) for row in result)),
  72. }
  73. format = "{0:{lengths[package]}} | {1:{lengths[version]}} | {2:{lengths[suite]}} | {3}"
  74. for row in result:
  75. yield format.format(
  76. row[t.c.package],
  77. row[t.c.version],
  78. row[t.c.display_suite],
  79. row[c_architectures],
  80. lengths=lengths,
  81. )
  82. elif format in ("control-suite", "heidi"):
  83. query = sql.select([t.c.package, t.c.version, t.c.architecture]).where(
  84. where
  85. )
  86. result = session.execute(query)
  87. for row in result:
  88. yield "{0} {1} {2}".format(
  89. row[t.c.package], row[t.c.version], row[t.c.architecture]
  90. )
  91. elif format == "python":
  92. c_architectures = sql.func.string_agg(
  93. t.c.architecture,
  94. pgsql.aggregate_order_by(
  95. ",", t.c.architecture_is_source.desc(), t.c.architecture
  96. ),
  97. )
  98. query = (
  99. sql.select(
  100. [
  101. t.c.package,
  102. t.c.version,
  103. t.c.display_suite,
  104. c_architectures,
  105. t.c.source,
  106. t.c.source_version,
  107. t.c.component,
  108. ]
  109. )
  110. .where(where)
  111. .group_by(
  112. t.c.package,
  113. t.c.version,
  114. t.c.display_suite,
  115. t.c.source,
  116. t.c.component,
  117. t.c.source_version,
  118. )
  119. )
  120. result = session.execute(query).fetchall()
  121. if len(result) == 0:
  122. return
  123. def val():
  124. return defaultdict(val)
  125. ret = val()
  126. for row in result:
  127. ret[row[t.c.package]][row[t.c.display_suite]][row[t.c.version]] = {
  128. "component": row[t.c.component],
  129. "architectures": row[c_architectures].split(","),
  130. "source": row[t.c.source],
  131. "source_version": row[t.c.source_version],
  132. }
  133. yield ret
  134. return
  135. else:
  136. raise ValueError("Unknown output format requested.")
  137. if highest is not None:
  138. query = (
  139. sql.select([t.c.package, sql.func.max(t.c.version)])
  140. .where(where)
  141. .group_by(t.c.package)
  142. .order_by(t.c.package)
  143. )
  144. result = session.execute(query)
  145. yield ""
  146. for row in result:
  147. yield "{0} ({1} {2})".format(row[0], highest, row[1])
  148. finally:
  149. session.close()