formatters.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. import json
  2. import logging
  3. import uuid
  4. from abc import ABCMeta, abstractmethod
  5. class BaseASFormatter(metaclass=ABCMeta):
  6. '''
  7. Format message as ActivityStreams.
  8. This is not really a `logging.Formatter` subclass, but a plain string
  9. builder.
  10. '''
  11. ev_names = {
  12. 'Update' : 'Resource Modification',
  13. 'Create' : 'Resource Creation',
  14. 'Delete' : 'Resource Deletion',
  15. }
  16. def __init__(self, uri, ev_type, time, type, data=None,
  17. data_fmt='text/turtle', metadata=None):
  18. '''
  19. Format output according to granularity level.
  20. NOTE: Granularity level does not refer to the logging levels, i.e.
  21. *when* a message gets logged, in fact all the Messaging logger messages
  22. are logged under the same level. This it is rather about *what* gets
  23. logged in a message.
  24. @param record (dict) This holds a dict with the following keys:
  25. - `uri`: URI of the resource.
  26. - `ev_type`: one of `create`, `delete` or `update`
  27. - `time`: Timestamp of the ev_type.
  28. - `data`: if messaging is configured with `provenance` level, this is
  29. a `rdflib.Graph` containing the triples that have been removed or
  30. added.
  31. - `metadata`: provenance metadata as a rdflib.Graph object. This
  32. contains properties such as actor(s), action (add/remove), etc. This is
  33. only meaningful for `ASDeltaFormatter`.
  34. '''
  35. self.uri = uri
  36. self.ev_type = ev_type
  37. self.time = time
  38. self.type = type
  39. self.data = data.serialize(format=data_fmt).decode('utf8') \
  40. if data else None
  41. self.metadata = metadata
  42. @abstractmethod
  43. def __str__(self):
  44. pass
  45. class ASResourceFormatter(BaseASFormatter):
  46. '''
  47. Sends information about a resource being created, updated or deleted, by
  48. who and when, with no further information about what changed.
  49. '''
  50. def __str__(self):
  51. '''
  52. Output structured data as string.
  53. '''
  54. ret = {
  55. '@context': 'https://www.w3.org/ns/activitystreams',
  56. 'id' : 'urn:uuid:{}'.format(uuid.uuid4()),
  57. 'type' : self.ev_type,
  58. 'name' : self.ev_names[self.ev_type],
  59. 'object' : {
  60. 'id' : self.uri,
  61. 'updated' : self.time,
  62. 'type' : self.type,
  63. },
  64. 'actor' : self.metadata.setdefault('actor', None),
  65. }
  66. return json.dumps(ret)
  67. class ASDeltaFormatter(BaseASFormatter):
  68. '''
  69. Sends the same information as `ASResourceFormatter` with the addition of
  70. the triples that were added and the ones that were removed in the request.
  71. This may be used to send rich provenance data to a preservation system.
  72. '''
  73. def __str__(self):
  74. '''
  75. Output structured data as string.
  76. '''
  77. ret = {
  78. '@context': 'https://www.w3.org/ns/activitystreams',
  79. 'id' : 'urn:uuid:{}'.format(uuid.uuid4()),
  80. 'type' : self.ev_type,
  81. 'name' : self.ev_names[self.ev_type],
  82. 'object' : {
  83. 'id' : self.uri,
  84. 'updated' : self.time,
  85. 'type' : self.type,
  86. },
  87. 'actor' : self.metadata.setdefault('actor', None),
  88. 'data' : self.data,
  89. }
  90. return json.dumps(ret)