module Braid class Mirror ATTRIBUTES = %w(url branch squashed revision lock remote_path) class UnknownType < BraidError def message "unknown type: #{super}" end end class PathRequired < BraidError def message 'path is required' end end include Operations::VersionControl attr_reader :path, :attributes def initialize(path, attributes = {}) @path = path.sub(/\/$/, '') @attributes = attributes end def self.new_from_options(url, options = {}) url = url.sub(/\/$/, '') branch = options['branch'] || 'master' path = (options['path'] || extract_path_from_url(url)).sub(/\/$/, '') raise PathRequired unless path squashed = !options['full'] remote_path = options['remote_path'] attributes = {'url' => url, 'branch' => branch, 'squashed' => squashed, 'remote_path' => remote_path} self.new(path, attributes) end def ==(comparison) path == comparison.path && attributes == comparison.attributes end def locked? !!lock end def squashed? !!squashed end def merged?(commit) # tip from spearce in #git: # `test z$(git merge-base A B) = z$(git rev-parse --verify A)` commit = git.rev_parse(commit) if squashed? !!base_revision && git.merge_base(commit, base_revision) == commit else git.merge_base(commit, 'HEAD') == commit end end def versioned_path(revision) "#{revision}:#{self.remote_path}" end def diff fetch remote_hash = git.rev_parse(versioned_path(base_revision)) local_hash = git.tree_hash(path) remote_hash != local_hash ? git.diff_tree(remote_hash, local_hash) : '' end def fetch git_cache.fetch(url) if cached? git.fetch(remote) end def cached? git.remote_url(remote) == cached_url end def base_revision if revision git.rev_parse(revision) else inferred_revision end end def cached_url git_cache.path(url) end def remote "#{branch}/braid/#{path}" end private def method_missing(name, *args) if ATTRIBUTES.find { |attribute| name.to_s =~ /^(#{attribute})(=)?$/ } if $2 attributes[$1] = args[0] else attributes[$1] end else raise NameError, "unknown attribute `#{name}'" end end def inferred_revision local_commits = git.rev_list('HEAD', "-- #{path}").split("\n") remote_hashes = git.rev_list("--pretty=format:\"%T\"", remote).split('commit ').map do |chunk| chunk.split("\n", 2).map { |value| value.strip } end hash = nil local_commits.each do |local_commit| local_tree = git.tree_hash(path, local_commit) match = remote_hashes.find { |_, remote_tree| local_tree == remote_tree } if match hash = match[0] break end end hash end def self.extract_path_from_url(url) return nil unless url name = File.basename(url) if File.extname(name) == '.git' # strip .git name[0..-5] else name end end end end