A fork of hyde, the static site generation. Some patches will be pushed upstream.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

188 lines
5.9 KiB

  1. # -*- coding: utf-8 -*-
  2. """
  3. Contains definition for a plugin protocol and other utiltities.
  4. """
  5. import abc
  6. from hyde import loader
  7. from hyde.util import getLoggerWithNullHandler
  8. from functools import partial
  9. logger = getLoggerWithNullHandler('hyde.engine')
  10. class PluginProxy(object):
  11. """
  12. A proxy class to raise events in registered plugins
  13. """
  14. def __init__(self, site):
  15. super(PluginProxy, self).__init__()
  16. self.site = site
  17. def __getattr__(self, method_name):
  18. if hasattr(Plugin, method_name):
  19. def __call_plugins__(*args):
  20. # logger.debug("Calling plugin method [%s]", method_name)
  21. res = None
  22. if self.site.plugins:
  23. for plugin in self.site.plugins:
  24. if hasattr(plugin, method_name):
  25. # logger.debug(
  26. # "\tCalling plugin [%s]",
  27. # plugin.__class__.__name__)
  28. function = getattr(plugin, method_name)
  29. res = function(*args)
  30. if res:
  31. targs = list(args)
  32. last = None
  33. if len(targs):
  34. last = targs.pop()
  35. targs.append(res if res else last)
  36. args = tuple(targs)
  37. return res
  38. return __call_plugins__
  39. raise HydeException(
  40. "Unknown plugin method [%s] called." % method_name)
  41. class Plugin(object):
  42. """
  43. The plugin protocol
  44. """
  45. __metaclass__ = abc.ABCMeta
  46. def __init__(self, site):
  47. super(Plugin, self).__init__()
  48. self.site = site
  49. self.logger = getLoggerWithNullHandler(self.__class__.__name__)
  50. def template_loaded(self, template):
  51. """
  52. Called when the template for the site has been identified.
  53. Handles the template loaded event to keep
  54. a reference to the template object.
  55. """
  56. self.template = template
  57. def __getattribute__(self, name):
  58. """
  59. Syntactic sugar for template methods
  60. """
  61. if name.startswith('t_') and self.template:
  62. attr = name[2:]
  63. if hasattr(self.template, attr):
  64. return self.template[attr]
  65. elif attr.endswith('_close_tag'):
  66. tag = attr.replace('_close_tag', '')
  67. return partial(self.template.get_close_tag, tag)
  68. elif attr.endswith('_open_tag'):
  69. tag = attr.replace('_open_tag', '')
  70. return partial(self.template.get_open_tag, tag)
  71. return super(Plugin, self).__getattribute__(name)
  72. def begin_generation(self):
  73. """
  74. Called when generation is about to take place.
  75. """
  76. pass
  77. def begin_site(self):
  78. """
  79. Called when the site is loaded completely. This implies that all the
  80. nodes and resources have been identified and are accessible in the
  81. site variable.
  82. """
  83. pass
  84. def begin_node(self, node):
  85. """
  86. Called when a node is about to be processed for generation.
  87. This method is called only when the entire node is generated.
  88. """
  89. pass
  90. def begin_text_resource(self, resource, text):
  91. """
  92. Called when a text resource is about to be processed for generation.
  93. The `text` parameter contains the resource text at this point
  94. in its lifecycle. It is the text that has been loaded and any
  95. plugins that are higher in the order may have tampered with it.
  96. But the text has not been processed by the template yet. Note that
  97. the source file associated with the text resource may not be modifed
  98. by any plugins.
  99. If this function returns a value, it is used as the text for further
  100. processing.
  101. """
  102. return text
  103. def begin_binary_resource(self, resource):
  104. """
  105. Called when a binary resource is about to be processed for generation.
  106. Plugins are free to modify the contents of the file.
  107. """
  108. pass
  109. def text_resource_complete(self, resource, text):
  110. """
  111. Called when a resource has been processed by the template.
  112. The `text` parameter contains the resource text at this point
  113. in its lifecycle. It is the text that has been processed by the
  114. template and any plugins that are higher in the order may have
  115. tampered with it. Note that the source file associated with the
  116. text resource may not be modifed by any plugins.
  117. If this function returns a value, it is used as the text for further
  118. processing.
  119. """
  120. return text
  121. def binary_resource_complete(self, resource):
  122. """
  123. Called when a binary resource has already been processed.
  124. Plugins are free to modify the contents of the file.
  125. """
  126. pass
  127. def node_complete(self, node):
  128. """
  129. Called when all the resources in the node have been processed.
  130. This method is called only when the entire node is generated.
  131. """
  132. pass
  133. def site_complete(self):
  134. """
  135. Called when the entire site has been processed. This method is called
  136. only when the entire site is generated.
  137. """
  138. pass
  139. def generation_complete(self):
  140. """
  141. Called when generation is completed.
  142. """
  143. pass
  144. def raise_event(self, event_name):
  145. return getattr(Plugin.proxy, event_name)()
  146. @staticmethod
  147. def load_all(site):
  148. """
  149. Loads plugins based on the configuration. Assigns the plugins to
  150. 'site.plugins'
  151. """
  152. site.plugins = [loader.load_python_object(name)(site)
  153. for name in site.config.plugins]
  154. @staticmethod
  155. def get_proxy(site):
  156. return PluginProxy(site)