doc.rb 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. module Docs
  2. class Doc
  3. INDEX_FILENAME = 'index.json'
  4. DB_FILENAME = 'db.json'
  5. class << self
  6. include Instrumentable
  7. attr_accessor :name, :slug, :type, :release, :abstract, :links
  8. def inherited(subclass)
  9. subclass.type = type
  10. end
  11. def version(version = nil, &block)
  12. return @version unless block_given?
  13. klass = Class.new(self)
  14. klass.name = name
  15. klass.slug = slug
  16. klass.version = version
  17. klass.release = release
  18. klass.links = links
  19. klass.class_exec(&block)
  20. @versions ||= []
  21. @versions << klass
  22. klass
  23. end
  24. def version=(value)
  25. @version = value.to_s
  26. end
  27. def versions
  28. @versions.presence || [self]
  29. end
  30. def version?
  31. version.present?
  32. end
  33. def versioned?
  34. @versions.presence
  35. end
  36. def name
  37. @name || super.demodulize
  38. end
  39. def slug
  40. slug = @slug || default_slug || raise('slug is required')
  41. version? ? "#{slug}~#{version_slug}" : slug
  42. end
  43. def version_slug
  44. return if version.blank?
  45. slug = version.downcase
  46. slug.gsub! '+', 'p'
  47. slug.gsub! '#', 's'
  48. slug.gsub! %r{[^a-z0-9\_\.]}, '_'
  49. slug
  50. end
  51. def path
  52. slug
  53. end
  54. def index_path
  55. File.join path, INDEX_FILENAME
  56. end
  57. def db_path
  58. File.join path, DB_FILENAME
  59. end
  60. def as_json
  61. json = { name: name, slug: slug, type: type }
  62. json[:links] = links if links.present?
  63. json[:version] = version if version.present? || defined?(@version)
  64. json[:release] = release if release.present?
  65. json
  66. end
  67. def store_page(store, id)
  68. store.open(path) do
  69. if page = new.build_page(id) and store_page?(page)
  70. store.write page[:store_path], page[:output]
  71. true
  72. else
  73. false
  74. end
  75. end
  76. end
  77. def store_pages(store)
  78. index = EntryIndex.new
  79. pages = PageDb.new
  80. store.replace(path) do
  81. new.build_pages do |page|
  82. next unless store_page?(page)
  83. store.write page[:store_path], page[:output]
  84. index.add page[:entries]
  85. pages.add page[:path], page[:output]
  86. end
  87. if index.present?
  88. store_index(store, INDEX_FILENAME, index)
  89. store_index(store, DB_FILENAME, pages)
  90. true
  91. else
  92. false
  93. end
  94. end
  95. end
  96. private
  97. def default_slug
  98. return if name =~ /[^A-Za-z0-9_]/
  99. name.downcase
  100. end
  101. def store_page?(page)
  102. page[:entries].present?
  103. end
  104. def store_index(store, filename, index)
  105. old_json = store.read(filename) || '{}'
  106. new_json = index.to_json
  107. instrument "#{filename.remove('.json')}.doc", before: old_json, after: new_json
  108. store.write(filename, new_json)
  109. end
  110. end
  111. def initialize
  112. raise NotImplementedError, "#{self.class} is an abstract class and cannot be instantiated." if self.class.abstract
  113. end
  114. def build_page(id, &block)
  115. raise NotImplementedError
  116. end
  117. def build_pages(&block)
  118. raise NotImplementedError
  119. end
  120. end
  121. end