doc.rb 3.8 KB

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