appveyor-irc-notify.py 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. # coding=utf8
  2. # Copyright (C) 2015-2016 Christopher R. Wood
  3. # Copyright (c) 2018 The Tor Project
  4. # Copyright (c) 2018 isis agora lovecruft
  5. #
  6. # From: https://raw.githubusercontent.com/gridsync/gridsync/def54f8166089b733d166665fdabcad4cdc526d8/misc/irc-notify.py
  7. # and: https://github.com/gridsync/gridsync
  8. #
  9. # Modified by nexB on October 2016:
  10. # - rework the handling of environment variables.
  11. # - made the script use functions
  12. # - support only Appveyor loading its environment variable to craft IRC notices.
  13. #
  14. # Modified by isis agora lovecruft <isis@torproject.org> in 2018:
  15. # - Make IRC server configurable.
  16. # - Make bot IRC nick deterministic.
  17. # - Make bot join the channel rather than sending NOTICE messages externally.
  18. # - Fix a bug which always caused sys.exit() to be logged as a traceback.
  19. # - Actually reset the IRC colour codes after printing.
  20. #
  21. # Modified by Marcin Cieślak in 2018:
  22. # - Accept UTF-8
  23. # - only guess github URLs
  24. # - stop using ANSI colors
  25. #
  26. # Modified by teor in 2018:
  27. # - fix github provider detection ('gitHub' or 'gitHubEnterprise', apparently)
  28. # - make short commits 10 hexdigits long (that's what git does for tor)
  29. # This program is free software; you can redistribute it and/or modify it under the
  30. # terms of the GNU General Public License as published by the Free Software Foundation;
  31. # either version 2 of the License, or (at your option) any later version.
  32. #
  33. # This program is distributed in the hope that it will be useful, but WITHOUT ANY
  34. # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
  35. # PARTICULAR PURPOSE. See the GNU General Public License for more details.
  36. #
  37. # You should have received a copy of the GNU General Public License along with this
  38. # program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street,
  39. # Fifth Floor, Boston, MA 02110-1301 USA.
  40. """Simple AppVeyor IRC notification script.
  41. The first argument is an IRC server and port; the second is the channel. Other
  42. arguments passed to the script will be sent as notice messages content and any
  43. {var}-formatted environment variables will be expanded automatically, replaced
  44. with a corresponding Appveyor environment variable value. Use commas to
  45. delineate multiple messages.
  46. Example:
  47. export APPVEYOR_URL=https://ci.appveyor.com
  48. export APPVEYOR_PROJECT_NAME=tor
  49. export APPVEYOR_REPO_COMMIT_AUTHOR=isislovecruft
  50. export APPVEYOR_REPO_COMMIT_TIMESTAMP=2018-04-23
  51. export APPVEYOR_REPO_PROVIDER=gihub
  52. export APPVEYOR_REPO_BRANCH=repo_branch
  53. export APPVEYOR_PULL_REQUEST_TITLE=pull_request_title
  54. export APPVEYOR_BUILD_VERSION=1
  55. export APPVEYOR_REPO_COMMIT=22c95b72e29248dc4de9b85e590ee18f6f587de8
  56. export APPVEYOR_REPO_COMMIT_MESSAGE="some IRC test"
  57. export APPVEYOR_ACCOUNT_NAME=isislovecruft
  58. export APPVEYOR_PULL_REQUEST_NUMBER=pull_request_number
  59. export APPVEYOR_REPO_NAME=isislovecruft/tor
  60. python ./appveyor-irc-notify.py irc.oftc.net:6697 tor-ci '{repo_name} {repo_branch} {short_commit} - {repo_commit_author}: {repo_commit_message}','Build #{build_version} passed. Details: {build_url} | Commit: {commit_url}
  61. See also https://github.com/gridsync/gridsync/blob/master/appveyor.yml for examples
  62. in Appveyor's YAML:
  63. on_success:
  64. - "python scripts/test/appveyor-irc-notify.py irc.oftc.net:6697 tor-ci success
  65. on_failure:
  66. - "python scripts/test/appveyor-irc-notify.py irc.oftc.net:6697 tor-ci failure
  67. """
  68. from __future__ import print_function
  69. from __future__ import absolute_import
  70. import os
  71. import random
  72. import socket
  73. import ssl
  74. import sys
  75. import time
  76. def appveyor_vars():
  77. """
  78. Return a dict of key value carfted from appveyor environment variables.
  79. """
  80. vars = dict([
  81. (
  82. v.replace('APPVEYOR_', '').lower(),
  83. os.getenv(v, '').decode('utf-8')
  84. ) for v in [
  85. 'APPVEYOR_URL',
  86. 'APPVEYOR_REPO_COMMIT_MESSAGE_EXTENDED',
  87. 'APPVEYOR_REPO_BRANCH',
  88. 'APPVEYOR_REPO_COMMIT_AUTHOR',
  89. 'APPVEYOR_REPO_COMMIT_AUTHOR_EMAIL',
  90. 'APPVEYOR_REPO_COMMIT_TIMESTAMP',
  91. 'APPVEYOR_REPO_PROVIDER',
  92. 'APPVEYOR_PROJECT_NAME',
  93. 'APPVEYOR_PULL_REQUEST_TITLE',
  94. 'APPVEYOR_BUILD_VERSION',
  95. 'APPVEYOR_REPO_COMMIT',
  96. 'APPVEYOR_REPO_COMMIT_MESSAGE',
  97. 'APPVEYOR_ACCOUNT_NAME',
  98. 'APPVEYOR_PULL_REQUEST_NUMBER',
  99. 'APPVEYOR_REPO_NAME'
  100. ]
  101. ])
  102. BUILD_FMT = u'{url}/project/{account_name}/{project_name}/build/{build_version}'
  103. if vars["repo_provider"].lower().startswith('github'):
  104. COMMIT_FMT = u'https://{repo_provider}.com/{repo_name}/commit/{repo_commit}'
  105. vars.update(commit_url=COMMIT_FMT.format(**vars))
  106. vars.update(
  107. build_url=BUILD_FMT.format(**vars),
  108. short_commit=vars["repo_commit"][:10],
  109. )
  110. return vars
  111. def notify():
  112. """
  113. Send IRC notification
  114. """
  115. apvy_vars = appveyor_vars()
  116. server, port = sys.argv[1].rsplit(":", 1)
  117. channel = sys.argv[2]
  118. success = sys.argv[3] == "success"
  119. failure = sys.argv[3] == "failure"
  120. if success or failure:
  121. messages = []
  122. messages.append(u"{repo_name} {repo_branch} {short_commit} - {repo_commit_author}: {repo_commit_message}")
  123. if success:
  124. m = u"Build #{build_version} passed. Details: {build_url}"
  125. if failure:
  126. m = u"Build #{build_version} failed. Details: {build_url}"
  127. if "commit_url" in apvy_vars:
  128. m += " Commit: {commit_url}"
  129. messages.append(m)
  130. else:
  131. messages = sys.argv[3:]
  132. messages = ' '.join(messages)
  133. messages = messages.decode("utf-8").split(',')
  134. print(repr(apvy_vars))
  135. messages = [msg.format(**apvy_vars).strip() for msg in messages]
  136. irc_username = 'appveyor-ci'
  137. irc_nick = irc_username
  138. # establish connection
  139. irc_sock = ssl.wrap_socket(socket.socket(socket.AF_INET, socket.SOCK_STREAM))
  140. irc_sock.connect((socket.gethostbyname(server), int(port)))
  141. irc_sock.send('NICK {0}\r\nUSER {0} * 0 :{0}\r\n'.format(irc_username).encode())
  142. irc_sock.send('JOIN #{0}\r\n'.format(channel).encode())
  143. irc_file = irc_sock.makefile()
  144. while irc_file:
  145. line = irc_file.readline()
  146. print(line.rstrip())
  147. response = line.split()
  148. if response[0] == 'PING':
  149. irc_file.send('PONG {}\r\n'.format(response[1]).encode())
  150. elif response[1] == '433':
  151. irc_sock.send('NICK {}\r\n'.format(irc_nick).encode())
  152. elif response[1] == '001':
  153. time.sleep(5)
  154. # send notification
  155. for msg in messages:
  156. print(u'PRIVMSG #{} :{}'.format(channel, msg).encode("utf-8"))
  157. irc_sock.send(u'PRIVMSG #{} :{}\r\n'.format(channel, msg).encode("utf-8"))
  158. time.sleep(5)
  159. return
  160. if __name__ == '__main__':
  161. try:
  162. notify()
  163. except:
  164. import traceback
  165. print('ERROR: Failed to send notification: \n' + traceback.format_exc())