diff options
author | drbrain <drbrain@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2010-04-01 07:45:16 +0000 |
---|---|---|
committer | drbrain <drbrain@b2dd03c8-39d4-4d8f-98ff-823fe69b080e> | 2010-04-01 07:45:16 +0000 |
commit | 46580b51477355fece514573c88cb67030f4a502 (patch) | |
tree | 779c1a64466643461b3daa4cd9a3548b84f0fd55 /lib/rdoc/ri/store.rb | |
parent | 9b40cdfe8c973a061c5683ad78c283b9ddb8b2e9 (diff) |
Import RDoc 2.5
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@27147 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'lib/rdoc/ri/store.rb')
-rw-r--r-- | lib/rdoc/ri/store.rb | 248 |
1 files changed, 248 insertions, 0 deletions
diff --git a/lib/rdoc/ri/store.rb b/lib/rdoc/ri/store.rb new file mode 100644 index 0000000000..81ffb7e674 --- /dev/null +++ b/lib/rdoc/ri/store.rb @@ -0,0 +1,248 @@ +require 'rdoc/code_objects' +require 'fileutils' + +## +# A set of ri data. +# +# The store manages reading and writing ri data for a project (gem, path, +# etc.) and maintains a cache of methods, classes and ancestors in the +# store. + +class RDoc::RI::Store + + ## + # Path this store reads or writes + + attr_accessor :path + + ## + # Type of ri datastore this was loaded from. See RDoc::RI::Driver, + # RDoc::RI::Paths. + + attr_accessor :type + + attr_reader :cache + + ## + # Creates a new Store of +type+ that will load or save to +path+ + + def initialize path, type = nil + @type = type + @path = path + + @cache = { + :class_methods => {}, + :instance_methods => {}, + :attributes => {}, + :modules => [], + :ancestors => {}, + } + end + + ## + # Ancestors cache accessor. Maps a klass name to an Array of its ancestors + # in this store. If Foo in this store inherits from Object, Kernel won't be + # listed (it will be included from ruby's ri store). + + def ancestors + @cache[:ancestors] + end + + ## + # Attributes cache accessor. Maps a class to an Array of its attributes. + + def attributes + @cache[:attributes] + end + + ## + # Path to the cache file + + def cache_path + File.join @path, 'cache.ri' + end + + ## + # Path to the ri data for +klass_name+ + + def class_file klass_name + name = klass_name.split('::').last + File.join class_path(klass_name), "cdesc-#{name}.ri" + end + + ## + # Class methods cache accessor. Maps a class to an Array of its class + # methods (not full name). + + def class_methods + @cache[:class_methods] + end + + ## + # Path where data for +klass_name+ will be stored (methods or class data) + + def class_path klass_name + File.join @path, *klass_name.split('::') + end + + ## + # Friendly rendition of #path + + def friendly_path + case type + when :gem then + sep = Regexp.union(*['/', File::ALT_SEPARATOR].compact) + @path =~ /#{sep}doc#{sep}(.*?)#{sep}ri$/ + "gem #{$1}" + when :home then '~/.ri' + when :site then 'ruby site' + when :system then 'ruby core' + else @path + end + end + + def inspect # :nodoc: + "#<%s:0x%x %s %p>" % [self.class, object_id, @path, modules.sort] + end + + ## + # Instance methods cache accessor. Maps a class to an Array of its + # instance methods (not full name). + + def instance_methods + @cache[:instance_methods] + end + + ## + # Loads cache file for this store + + def load_cache + open cache_path, 'rb' do |io| + @cache = Marshal.load io.read + end + rescue Errno::ENOENT + end + + ## + # Loads ri data for +klass_name+ + + def load_class klass_name + open class_file(klass_name), 'rb' do |io| + Marshal.load io.read + end + end + + ## + # Loads ri data for +method_name+ in +klass_name+ + + def load_method klass_name, method_name + open method_file(klass_name, method_name), 'rb' do |io| + Marshal.load io.read + end + end + + ## + # Path to the ri data for +method_name+ in +klass_name+ + + 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 = if ''.respond_to? :ord then + method_name.gsub(/\W/) { "%%%02x" % $&[0].ord } + else + method_name.gsub(/\W/) { "%%%02x" % $&[0] } + end + + File.join class_path(klass_name), "#{method_name}-#{method_type}.ri" + end + + ## + # Modules cache accessor. An Array of all the modules (and classes) in the + # store. + + def modules + @cache[:modules] + end + + ## + # Writes the cache file for this store + + def save_cache + # HACK mongrel-1.1.5 documents its files twice + @cache[:attributes]. each do |_, m| m.uniq!; m.sort! end + @cache[:class_methods]. each do |_, m| m.uniq!; m.sort! end + @cache[:instance_methods].each do |_, m| m.uniq!; m.sort! end + + open cache_path, 'wb' do |io| + Marshal.dump @cache, io + end + end + + ## + # Writes the ri data for +klass+ + + def save_class klass + FileUtils.mkdir_p class_path(klass.full_name) + + @cache[:modules] << klass.full_name + + path = class_file klass.full_name + + begin + disk_klass = nil + + open path, 'rb' do |io| + disk_klass = Marshal.load io.read + end + + klass.merge disk_klass + rescue Errno::ENOENT + end + + # BasicObject has no ancestors + ancestors = klass.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][klass.full_name] ||= [] + @cache[:ancestors][klass.full_name].push(*ancestors) + + attributes = klass.attributes.map do |attribute| + "#{attribute.type} #{attribute.name}" + end + + unless attributes.empty? then + @cache[:attributes][klass.full_name] ||= [] + @cache[:attributes][klass.full_name].push(*attributes) + end + + open path, 'wb' do |io| + Marshal.dump klass, io + end + end + + ## + # Writes the ri data for +method+ on +klass+ + + def save_method klass, method + FileUtils.mkdir_p class_path(klass.full_name) + + cache = if method.singleton then + @cache[:class_methods] + else + @cache[:instance_methods] + end + cache[klass.full_name] ||= [] + cache[klass.full_name] << method.name + + open method_file(klass.full_name, method.full_name), 'wb' do |io| + Marshal.dump method, io + end + end + +end + |