dodo.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. import subprocess
  2. import os
  3. import sys
  4. from doit.tools import LongRunning
  5. from dodo_utils import wait_net_service, wait_pgsql_db, \
  6. up2date_hasimage, up2date_anyimages, \
  7. run_task_func, scan_dir
  8. COMPOSE = 'docker-compose -p feedati'
  9. DOIT_CONFIG = {'default_tasks': ['up'], 'backend': 'sqlite3'}
  10. def task_build():
  11. '''builda il container docker'''
  12. return {
  13. 'uptodate': [up2date_anyimages],
  14. 'file_dep': ['docker-compose.yml',
  15. 'rss-bridge/Dockerfile',
  16. 'panel/Dockerfile',
  17. ] + list(scan_dir('docker')),
  18. 'actions': [COMPOSE + ' build'],
  19. 'clean': [run_task_func(task__build_rm),
  20. run_task_func(task__build_rmi)],
  21. 'doc': '''
  22. This task recreates every docker container. While it is automatically run for most changes in the
  23. development environment, please remember that if you want to run it manually to grab changes in the
  24. docker hub, you need to run `doit run -a build`.
  25. '''
  26. }
  27. def task__build_rm():
  28. '''rimuove container avviati'''
  29. return {'actions': [
  30. "docker container ls -a --format '{{.ID}}\t{{.Names}}'|"
  31. "awk '$2 ~ /^feedati_/ { print $1 }' | "
  32. "xargs -r docker container rm",
  33. ]}
  34. def task__build_rmi():
  35. '''rimuove immagini ottenute con build'''
  36. return {
  37. 'actions': [
  38. r"docker images -q 'feedati/*' |"
  39. "xargs -r --verbose docker rmi",
  40. ]
  41. }
  42. def task__dbprepare_clean():
  43. '''rimuove il dump caricato sul db'''
  44. return {
  45. 'actions': [
  46. "docker ps -aqf name=feedati_db|xargs -r docker container rm ",
  47. "docker volume rm feedati_postgres_data || true",
  48. ]
  49. }
  50. def stop():
  51. subprocess.check_call((COMPOSE + ' stop').split())
  52. return True
  53. def task_dbprepare():
  54. '''applica il dump sql al container del db'''
  55. panelcli = 'docker-compose -p feedati exec -T panel ./panelcli -db-pass panelpass -db-addr db:5432 '
  56. psql = r'docker exec -i $(docker ps -aqf name=feedati_db) env PGPASSWORD=superpass psql -h 127.0.0.1 -f - -d feeds superadmin '
  57. return {
  58. 'setup': ['_dbprepare_clean', 'build'],
  59. 'file_dep': list(scan_dir('docker', '*.sql')),
  60. 'actions': [
  61. (COMPOSE + ' up -d db panel').split(),
  62. (wait_net_service, ('localhost', 5432, 300)),
  63. (wait_pgsql_db, ('feedati_db', 'feeds', 'superadmin')),
  64. 'echo LOADING DB',
  65. # dbconf.sql would be better put in /docker-entrypoint-initdb.d/init-user-db.sh
  66. # (see https://hub.docker.com/r/library/postgres/ )
  67. psql + ' < docker/dbconf.sql',
  68. psql + ' < docker/ttrss.sql',
  69. panelcli + 'setup',
  70. panelcli + 'adduser -password password admin ad@m.in',
  71. psql + ' < docker/dbconf-perms.sql',
  72. 'echo DB RESTORED',
  73. ],
  74. 'teardown': [(stop, [])],
  75. 'uptodate': [up2date_hasimage('feedati_postgres_data')()],
  76. 'clean': [run_task_func(task__dbprepare_clean)]
  77. }
  78. def task__fix_perms():
  79. '''fix permissions for shared www dir'''
  80. # currently empty, but keeping it for later use
  81. return {
  82. 'actions': None,
  83. }
  84. def task_up():
  85. '''RUN that stuff!'''
  86. return {
  87. 'task_dep': ['build', 'dbprepare', '_fix_perms'],
  88. 'teardown': [(stop, [])],
  89. 'actions': [LongRunning(
  90. (COMPOSE + ' up').split(),
  91. shell=False)
  92. ]
  93. }
  94. def _get_valid_services():
  95. srv = subprocess.check_output((COMPOSE + ' config --services').split())
  96. return srv.decode('utf-8').strip().split()
  97. def restart(services):
  98. '''This actually restart the container(s).'''
  99. valid = _get_valid_services()
  100. err = False
  101. for service in services:
  102. if service not in valid:
  103. print('ERROR: invalid service %s' % service, file=sys.stderr)
  104. err = True
  105. if err or not services:
  106. print('Valid services are: ' + ', '.join(valid), file=sys.stderr)
  107. return False
  108. for service in services:
  109. subprocess.check_call((COMPOSE + ' restart %s' % service).split())
  110. return True
  111. def task_restart():
  112. '''Restarts a container specified via commandline.'''
  113. return {
  114. 'params': [{'name': 'services',
  115. 'short': 's',
  116. 'long': 'service',
  117. 'type': list,
  118. 'default': [],
  119. 'help': "the list of services to be restarted"}],
  120. 'actions': [(restart, )],
  121. }
  122. def task__cleanall():
  123. '''clean everything there is to clean'''
  124. return {
  125. 'task_dep': ['_build_rm', '_build_rmi', '_dbprepare_clean'],
  126. 'actions': None
  127. }