A set of rdoc data for a single project (gem, path, etc.).
The store manages reading and writing ri data for a project and maintains a cache of methods, classes and ancestors in the store.
The store maintains a cache
of its contents for faster lookup. After adding items to the store it must be flushed using save_cache
. The cache contains the following structures:
@cache = {
:ancestors => {}, # class name => ancestor names
:attributes => {}, # class name => attributes
:class_methods => {}, # class name => class methods
:instance_methods => {}, # class name => instance methods
:modules => [], # classes and modules in this store
:pages => [], # page names
}
- CLASS RDoc::Store::Error
- CLASS RDoc::Store::MissingFileError
- A
- C
- F
- I
- L
- M
- N
- P
- R
- S
- T
- U
Attributes
[R] | c_class_variables | Maps C variables to class or module names for each parsed C file. |
[R] | c_singleton_class_variables | Maps C variables to singleton class names for each parsed C file. |
[R] | cache | The contents of the |
[RW] | dry_run | If true this |
[RW] | encoding | The encoding of the contents in the |
[RW] | path | Path this store reads or writes |
[RW] | rdoc | The |
[RW] | type | Type of ri datastore this was loaded from. See RDoc::RI::Driver, RDoc::RI::Paths. |
[R] | unmatched_constant_alias | The lazy constants alias will be discovered in passing |
Class Public methods
new(path = nil, type = nil) Link
Creates a new Store
of type
that will load or save to path
# File ruby/lib/rdoc/store.rb, line 127 def initialize path = nil, type = nil @dry_run = false @encoding = nil @path = path @rdoc = nil @type = type @cache = { :ancestors => {}, :attributes => {}, :class_methods => {}, :c_class_variables => {}, :c_singleton_class_variables => {}, :encoding => @encoding, :instance_methods => {}, :main => nil, :modules => [], :pages => [], :title => nil, } @classes_hash = {} @modules_hash = {} @files_hash = {} @text_files_hash = {} @c_enclosure_classes = {} @c_enclosure_names = {} @c_class_variables = {} @c_singleton_class_variables = {} @unique_classes = nil @unique_modules = nil @unmatched_constant_alias = {} end
Instance Public methods
add_c_enclosure(variable, namespace) Link
Adds module
as an enclosure (namespace) for the given variable
for C files.
add_c_variables(c_parser) Link
Adds C variables from an RDoc::Parser::C
add_file(absolute_name, relative_name: absolute_name, parser: nil) Link
Adds the file with name
as an RDoc::TopLevel
to the store. Returns the created RDoc::TopLevel
.
# File ruby/lib/rdoc/store.rb, line 188 def add_file absolute_name, relative_name: absolute_name, parser: nil unless top_level = @files_hash[relative_name] then top_level = RDoc::TopLevel.new absolute_name, relative_name top_level.parser = parser if parser top_level.store = self @files_hash[relative_name] = top_level @text_files_hash[relative_name] = top_level if top_level.text? end top_level end
ancestors() Link
cache_path() Link
Path to the cache file
class_file(klass_name) Link
Path to the ri data for klass_name
class_methods() Link
class_path(klass_name) Link
Path where data for klass_name
will be stored (methods or class data)
classes_hash() Link
complete(min_visibility) Link
Prepares the RDoc
code object tree for use by a generator.
It finds unique classes/modules defined, and replaces classes/modules that are aliases for another one by a copy with RDoc::ClassModule#is_alias_for
set.
It updates the RDoc::ClassModule#constant_aliases
attribute of “real” classes or modules.
It also completely removes the classes and modules that should be removed from the documentation and the methods that have a visibility below min_visibility
, which is the --visibility
option.
# File ruby/lib/rdoc/store.rb, line 322 def complete min_visibility fix_basic_object_inheritance # cache included modules before they are removed from the documentation all_classes_and_modules.each { |cm| cm.ancestors } unless min_visibility == :nodoc then remove_nodoc @classes_hash remove_nodoc @modules_hash end @unique_classes = find_unique @classes_hash @unique_modules = find_unique @modules_hash unique_classes_and_modules.each do |cm| cm.complete min_visibility end @files_hash.each_key do |file_name| tl = @files_hash[file_name] unless tl.text? then tl.modules_hash.clear tl.classes_hash.clear tl.classes_or_modules.each do |cm| name = cm.full_name if cm.type == 'class' then tl.classes_hash[name] = cm if @classes_hash[name] else tl.modules_hash[name] = cm if @modules_hash[name] end end end end end
files_hash() Link
find_c_enclosure(variable) Link
Finds the enclosure (namespace) for the given C variable
.
# File ruby/lib/rdoc/store.rb, line 369 def find_c_enclosure variable @c_enclosure_classes.fetch variable do break unless name = @c_enclosure_names[variable] mod = find_class_or_module name unless mod then loaded_mod = load_class_data name file = loaded_mod.in_files.first return unless file # legacy data source file.store = self mod = file.add_module RDoc::NormalModule, name end @c_enclosure_classes[variable] = mod end end
find_class_named(name) Link
Finds the class with name
in all discovered classes
find_class_named_from(name, from) Link
Finds the class with name
starting in namespace from
# File ruby/lib/rdoc/store.rb, line 401 def find_class_named_from name, from from = find_class_named from unless RDoc::Context === from until RDoc::TopLevel === from do return nil unless from klass = from.find_class_named name return klass if klass from = from.parent end find_class_named name end
find_class_or_module(name) Link
Finds the class or module with name
find_file_named(name) Link
Finds the file with name
in all discovered files
find_module_named(name) Link
Finds the module with name
in all discovered modules
find_text_page(file_name) Link
Returns the RDoc::TopLevel
that is a text file and has the given file_name
find_unique(all_hash) Link
Finds unique classes/modules defined in all_hash
, and returns them as an array. Performs the alias updates in all_hash
: see ::complete.
fix_basic_object_inheritance() Link
Fixes the erroneous BasicObject < Object
in 1.9.
Because we assumed all classes without a stated superclass inherit from Object
, we have the above wrong inheritance.
We fix BasicObject
right away if we are running in a Ruby version >= 1.9.
instance_methods() Link
Instance methods cache accessor. Maps a class to an Array
of its instance methods (not full name).
load_all() Link
Loads all items from this store into memory. This recreates a documentation tree for use by a generator
# File ruby/lib/rdoc/store.rb, line 511 def load_all load_cache module_names.each do |module_name| mod = find_class_or_module(module_name) || load_class(module_name) # load method documentation since the loaded class/module does not have # it loaded_methods = mod.method_list.map do |method| load_method module_name, method.full_name end mod.method_list.replace loaded_methods loaded_attributes = mod.attributes.map do |attribute| load_method module_name, attribute.full_name end mod.attributes.replace loaded_attributes end all_classes_and_modules.each do |mod| descendent_re = /^#{mod.full_name}::[^:]+$/ module_names.each do |name| next unless name =~ descendent_re descendent = find_class_or_module name case descendent when RDoc::NormalClass then mod.classes_hash[name] = descendent when RDoc::NormalModule then mod.modules_hash[name] = descendent end end end @cache[:pages].each do |page_name| page = load_page page_name @files_hash[page_name] = page @text_files_hash[page_name] = page if page.text? end end
load_cache() Link
Loads cache file for this store
# File ruby/lib/rdoc/store.rb, line 559 def load_cache #orig_enc = @encoding @cache = marshal_load(cache_path) load_enc = @cache[:encoding] # TODO this feature will be time-consuming to add: # a) Encodings may be incompatible but transcodeable # b) Need to warn in the appropriate spots, wherever they may be # c) Need to handle cross-cache differences in encodings # d) Need to warn when generating into a cache with different encodings # #if orig_enc and load_enc != orig_enc then # warn "Cached encoding #{load_enc} is incompatible with #{orig_enc}\n" \ # "from #{path}/cache.ri" unless # Encoding.compatible? orig_enc, load_enc #end @encoding = load_enc unless @encoding @cache[:pages] ||= [] @cache[:main] ||= nil @cache[:c_class_variables] ||= {} @cache[:c_singleton_class_variables] ||= {} @cache[:c_class_variables].each do |_, map| map.each do |variable, name| @c_enclosure_names[variable] = name end end @cache rescue Errno::ENOENT end
load_class(klass_name) Link
Loads ri data for klass_name
and hooks it up to this store.
# File ruby/lib/rdoc/store.rb, line 598 def load_class klass_name obj = load_class_data klass_name obj.store = self case obj when RDoc::NormalClass then @classes_hash[klass_name] = obj when RDoc::SingleClass then @classes_hash[klass_name] = obj when RDoc::NormalModule then @modules_hash[klass_name] = obj end end
load_class_data(klass_name) Link
Loads ri data for klass_name
load_method(klass_name, method_name) Link
Loads ri data for method_name
in klass_name
# File ruby/lib/rdoc/store.rb, line 629 def load_method klass_name, method_name file = method_file klass_name, method_name obj = marshal_load(file) obj.store = self obj.parent ||= find_class_or_module(klass_name) || load_class(klass_name) obj rescue Errno::ENOENT => e error = MissingFileError.new(self, file, klass_name + method_name) error.set_backtrace e.backtrace raise error end
load_page(page_name) Link
Loads ri data for page_name
main() Link
make_variable_map(variables) Link
Converts the variable => ClassModule map variables
from a C parser into a variable => class name map.
method_file(klass_name, method_name) Link
Path to the ri data for method_name
in klass_name
# File ruby/lib/rdoc/store.rb, line 689 def method_file klass_name, method_name method_name = method_name.split('::').last method_name =~ /#(.*)/ method_type = $1 ? 'i' : 'c' method_name = $1 if $1 method_name = method_name.gsub(/\W/) { "%%%02x" % $&[0].ord } File.join class_path(klass_name), "#{method_name}-#{method_type}.ri" end
module_names() Link
Modules cache accessor. An Array
of all the module (and class) names in the store.
modules_hash() Link
page(name) Link
Returns the RDoc::TopLevel
that is a text file and has the given name
page_file(page_name) Link
Path to the ri data for page_name
remove_nodoc(all_hash) Link
Removes from all_hash
the contexts that are nodoc or have no content.
save() Link
Saves all entries in the store
# File ruby/lib/rdoc/store.rb, line 747 def save load_cache all_classes_and_modules.each do |klass| save_class klass klass.each_method do |method| save_method klass, method end klass.each_attribute do |attribute| save_method klass, attribute end end all_files.each do |file| save_page file end save_cache end
save_cache() Link
Writes the cache file for this store
# File ruby/lib/rdoc/store.rb, line 772 def save_cache clean_cache_collection @cache[:ancestors] clean_cache_collection @cache[:attributes] clean_cache_collection @cache[:class_methods] clean_cache_collection @cache[:instance_methods] @cache[:modules].uniq! @cache[:modules].sort! @cache[:pages].uniq! @cache[:pages].sort! @cache[:encoding] = @encoding # this gets set twice due to assert_cache @cache[:c_class_variables].merge! @c_class_variables @cache[:c_singleton_class_variables].merge! @c_singleton_class_variables return if @dry_run File.open cache_path, 'wb' do |io| Marshal.dump @cache, io end end
save_class(klass) Link
Writes the ri data for klass
(or module)
# File ruby/lib/rdoc/store.rb, line 799 def save_class klass full_name = klass.full_name FileUtils.mkdir_p class_path(full_name) unless @dry_run @cache[:modules] << full_name path = class_file full_name begin disk_klass = load_class full_name klass = disk_klass.merge klass rescue MissingFileError end # BasicObject has no ancestors ancestors = klass.direct_ancestors.compact.map do |ancestor| # HACK for classes we don't know about (class X < RuntimeError) String === ancestor ? ancestor : ancestor.full_name end @cache[:ancestors][full_name] ||= [] @cache[:ancestors][full_name].concat ancestors attribute_definitions = klass.attributes.map do |attribute| "#{attribute.definition} #{attribute.name}" end unless attribute_definitions.empty? then @cache[:attributes][full_name] ||= [] @cache[:attributes][full_name].concat attribute_definitions end to_delete = [] unless klass.method_list.empty? then @cache[:class_methods][full_name] ||= [] @cache[:instance_methods][full_name] ||= [] class_methods, instance_methods = klass.method_list.partition { |meth| meth.singleton } class_methods = class_methods. map { |method| method.name } instance_methods = instance_methods.map { |method| method.name } attribute_names = klass.attributes.map { |attr| attr.name } old = @cache[:class_methods][full_name] - class_methods to_delete.concat old.map { |method| method_file full_name, "#{full_name}::#{method}" } old = @cache[:instance_methods][full_name] - instance_methods - attribute_names to_delete.concat old.map { |method| method_file full_name, "#{full_name}##{method}" } @cache[:class_methods][full_name] = class_methods @cache[:instance_methods][full_name] = instance_methods end return if @dry_run FileUtils.rm_f to_delete File.open path, 'wb' do |io| Marshal.dump klass, io end end
save_method(klass, method) Link
Writes the ri data for method
on klass
# File ruby/lib/rdoc/store.rb, line 873 def save_method klass, method full_name = klass.full_name FileUtils.mkdir_p class_path(full_name) unless @dry_run cache = if method.singleton then @cache[:class_methods] else @cache[:instance_methods] end cache[full_name] ||= [] cache[full_name] << method.name return if @dry_run File.open method_file(full_name, method.full_name), 'wb' do |io| Marshal.dump method, io end end
save_page(page) Link
Writes the ri data for page
# File ruby/lib/rdoc/store.rb, line 896 def save_page page return unless page.text? path = page_file page.full_name FileUtils.mkdir_p File.dirname(path) unless @dry_run cache[:pages] ||= [] cache[:pages] << page.full_name return if @dry_run File.open path, 'wb' do |io| Marshal.dump page, io end end
source() Link
Source of the contents of this store.
For a store from a gem the source is the gem name. For a store from the home directory the source is “home”. For system ri store (the standard library documentation) the source is“ruby”. For a store from the site ri directory the store is “site”. For other stores the source is the path
.
title() Link
unique_classes() Link
Returns the unique classes discovered by RDoc
.
::complete must have been called prior to using this method.
unique_classes_and_modules() Link
Returns the unique classes and modules discovered by RDoc
. ::complete must have been called prior to using this method.
unique_modules() Link
Returns the unique modules discovered by RDoc
. ::complete must have been called prior to using this method.
update_parser_of_file(absolute_name, parser) Link
Sets the parser of absolute_name
, unless it from a source code file.