formatters.py 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. import json
  2. import logging
  3. import uuid
  4. from activipy import vocab
  5. class ActivityStreamsFormatter:
  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 present with messaging level set to `provenance`.
  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. def __str__(self):
  43. '''
  44. Output structured data as string.
  45. '''
  46. ret = {
  47. '@context': 'https://www.w3.org/ns/activitystreams',
  48. 'id' : 'urn:uuid:{}'.format(uuid.uuid4()),
  49. 'type' : self.ev_type,
  50. 'name' : self.ev_names[self.ev_type],
  51. 'object' : {
  52. 'id' : self.uri,
  53. 'updated' : self.time,
  54. 'type' : self.type,
  55. },
  56. 'actor' : self.metadata.setdefault('actor', None),
  57. 'data' : self.data or '',
  58. }
  59. return json.dumps(ret)