04-retain-check-source-persist-diff-port.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #!/usr/bin/env python3
  2. # Test for CVE-2018-12546, with the broker being stopped to write the persistence file, plus subscriber on different port.
  3. from mosq_test_helper import *
  4. import os.path
  5. import signal
  6. def write_config(filename, port1, port2, per_listener):
  7. with open(filename, 'w') as f:
  8. f.write("per_listener_settings %s\n" % (per_listener))
  9. f.write("check_retain_source true\n")
  10. f.write("port %d\n" % (port1))
  11. f.write("allow_anonymous true\n")
  12. f.write("acl_file %s\n" % (filename.replace('.conf', '.acl')))
  13. f.write("persistence true\n")
  14. f.write("persistence_file %s\n" % (filename.replace('.conf', '.db')))
  15. f.write("listener %d\n" % (port2))
  16. f.write("allow_anonymous true\n")
  17. def write_acl_1(filename, username):
  18. with open(filename, 'w') as f:
  19. if username is not None:
  20. f.write('user %s\n' % (username))
  21. f.write('topic readwrite test/topic\n')
  22. def write_acl_2(filename, username):
  23. with open(filename, 'w') as f:
  24. if username is not None:
  25. f.write('user %s\n' % (username))
  26. f.write('topic read test/topic\n')
  27. def do_test(proto_ver, per_listener, username):
  28. conf_file = os.path.basename(__file__).replace('.py', '.conf')
  29. write_config(conf_file, port1, port2, per_listener)
  30. persistence_file = os.path.basename(__file__).replace('.py', '.db')
  31. try:
  32. os.remove(persistence_file)
  33. except OSError:
  34. pass
  35. acl_file = os.path.basename(__file__).replace('.py', '.acl')
  36. write_acl_1(acl_file, username)
  37. rc = 1
  38. keepalive = 60
  39. connect_packet = mosq_test.gen_connect("retain-check", keepalive=keepalive, username=username, proto_ver=proto_ver)
  40. connack_packet = mosq_test.gen_connack(rc=0, proto_ver=proto_ver)
  41. if per_listener == "true":
  42. u = None
  43. else:
  44. # If per listener is false, then the second client will be denied
  45. # unless we provide a username
  46. u = username
  47. connect2_packet = mosq_test.gen_connect("retain-recv", keepalive=keepalive, username=u, proto_ver=proto_ver)
  48. connack2_packet = mosq_test.gen_connack(rc=0, proto_ver=proto_ver)
  49. mid = 1
  50. publish_packet = mosq_test.gen_publish("test/topic", qos=0, payload="retained message", retain=True, proto_ver=proto_ver)
  51. subscribe_packet = mosq_test.gen_subscribe(mid, "test/topic", 0, proto_ver=proto_ver)
  52. suback_packet = mosq_test.gen_suback(mid, 0, proto_ver=proto_ver)
  53. broker = mosq_test.start_broker(filename=os.path.basename(__file__), use_conf=True, port=port1)
  54. try:
  55. sock = mosq_test.do_client_connect(connect_packet, connack_packet, port=port1)
  56. sock.send(publish_packet)
  57. sock.close()
  58. sock = mosq_test.do_client_connect(connect2_packet, connack2_packet, port=port2)
  59. mosq_test.do_send_receive(sock, subscribe_packet, suback_packet, "suback 1")
  60. mosq_test.expect_packet(sock, "publish", publish_packet)
  61. sock.close()
  62. # Remove "write" ability
  63. write_acl_2(acl_file, username)
  64. broker.terminate()
  65. broker.wait()
  66. if os.path.isfile(persistence_file) == False:
  67. raise FileNotFoundError("Persistence file not written")
  68. broker = mosq_test.start_broker(filename=os.path.basename(__file__), use_conf=True, port=port1)
  69. sock = mosq_test.do_client_connect(connect2_packet, connack2_packet, port=port2)
  70. mosq_test.do_send_receive(sock, subscribe_packet, suback_packet, "suback 2")
  71. # If we receive the retained message here, it is a failure.
  72. mosq_test.do_ping(sock)
  73. rc = 0
  74. sock.close()
  75. except mosq_test.TestError:
  76. pass
  77. finally:
  78. broker.terminate()
  79. broker.wait()
  80. os.remove(conf_file)
  81. os.remove(acl_file)
  82. try:
  83. os.remove(persistence_file)
  84. except FileNotFoundError:
  85. pass
  86. (stdo, stde) = broker.communicate()
  87. if rc:
  88. print(stde.decode('utf-8'))
  89. exit(rc)
  90. (port1, port2) = mosq_test.get_port(2)
  91. do_test(proto_ver=4, per_listener="true", username=None)
  92. do_test(proto_ver=4, per_listener="true", username="test")
  93. do_test(proto_ver=4, per_listener="false", username=None)
  94. do_test(proto_ver=4, per_listener="false", username="test")
  95. do_test(proto_ver=5, per_listener="true", username=None)
  96. do_test(proto_ver=5, per_listener="true", username="test")
  97. do_test(proto_ver=5, per_listener="false", username=None)
  98. do_test(proto_ver=5, per_listener="false", username="test")