homesick-1.1.3/0000755000004100000410000000000012642331233013345 5ustar www-datawww-datahomesick-1.1.3/Rakefile0000644000004100000410000000352012642331233015012 0ustar www-datawww-datarequire 'rubygems' require 'bundler' require_relative 'lib/homesick/version' begin Bundler.setup(:default, :development) rescue Bundler::BundlerError => e $stderr.puts e.message $stderr.puts "Run `bundle install` to install missing gems" exit e.status_code end require 'rake' require 'jeweler' Jeweler::Tasks.new do |gem| gem.name = "homesick" gem.summary = %Q{Your home directory is your castle. Don't leave your dotfiles behind.} gem.description = %Q{ Your home directory is your castle. Don't leave your dotfiles behind. Homesick is sorta like rip, but for dotfiles. It uses git to clone a repository containing dotfiles, and saves them in ~/.homesick. It then allows you to symlink all the dotfiles into place with a single command. } gem.email = ["josh@technicalpickles.com", "info@muratayusuke.com"] gem.homepage = "http://github.com/technicalpickles/homesick" gem.authors = ["Joshua Nichols", "Yusuke Murata"] gem.version = Homesick::Version::STRING gem.license = "MIT" # Have dependencies? Add them to Gemfile # gem is a Gem::Specification... see http://www.rubygems.org/read/chapter/20 for additional settings end Jeweler::GemcutterTasks.new require 'rspec/core/rake_task' RSpec::Core::RakeTask.new(:spec) do |spec| spec.pattern = FileList['spec/**/*_spec.rb'] end RSpec::Core::RakeTask.new(:rcov) do |spec| spec.pattern = 'spec/**/*_spec.rb' spec.rcov = true end task :rubocop do if RUBY_VERSION >= '1.9.2' system('rubocop') end end task :test do Rake::Task['spec'].execute Rake::Task['rubocop'].execute end task :default => :test require 'rdoc/task' Rake::RDocTask.new do |rdoc| version = File.exist?('VERSION') ? File.read('VERSION') : "" rdoc.rdoc_dir = 'rdoc' rdoc.title = "homesick #{version}" rdoc.rdoc_files.include('README*') rdoc.rdoc_files.include('lib/**/*.rb') end homesick-1.1.3/bin/0000755000004100000410000000000012642331233014115 5ustar www-datawww-datahomesick-1.1.3/bin/homesick0000755000004100000410000000026012642331233015643 0ustar www-datawww-data#!/usr/bin/env ruby require 'pathname' lib = Pathname.new(__FILE__).dirname.join('..', 'lib').expand_path $LOAD_PATH.unshift lib.to_s require 'homesick' Homesick::CLI.start homesick-1.1.3/Gemfile0000644000004100000410000000136112642331233014641 0ustar www-datawww-datarequire 'rbconfig' source 'https://rubygems.org' # Add dependencies required to use your gem here. gem "thor", ">= 0.14.0" # Add dependencies to develop your gem here. # Include everything needed to run rake, tests, features, etc. group :development do gem "rake", ">= 0.8.7" gem "rspec", "~> 3.1.0" gem "guard" gem "guard-rspec" gem "rb-readline", "~> 0.5.0" gem "jeweler", ">= 1.6.2" gem 'coveralls', require: false gem "test_construct" gem "capture-output", "~> 1.0.0" if RbConfig::CONFIG['host_os'] =~ /linux|freebsd|openbsd|sunos|solaris/ gem 'libnotify' end if RbConfig::CONFIG['host_os'] =~ /darwin|mac os/ gem 'terminal-notifier-guard', '~> 1.6.1' end if RUBY_VERSION >= '1.9.2' gem "rubocop" end end homesick-1.1.3/homesick.gemspec0000644000004100000410000000706412642331233016523 0ustar www-datawww-data# Generated by jeweler # DO NOT EDIT THIS FILE DIRECTLY # Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec' # -*- encoding: utf-8 -*- # stub: homesick 1.1.3 ruby lib Gem::Specification.new do |s| s.name = "homesick" s.version = "1.1.3" s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version= s.require_paths = ["lib"] s.authors = ["Joshua Nichols", "Yusuke Murata"] s.date = "2015-10-31" s.description = "\n Your home directory is your castle. Don't leave your dotfiles behind.\n \n\n Homesick is sorta like rip, but for dotfiles. It uses git to clone a repository containing dotfiles, and saves them in ~/.homesick. It then allows you to symlink all the dotfiles into place with a single command. \n\n " s.email = ["josh@technicalpickles.com", "info@muratayusuke.com"] s.executables = ["homesick"] s.extra_rdoc_files = [ "ChangeLog.markdown", "LICENSE", "README.markdown" ] s.files = [ ".document", ".rspec", ".rubocop.yml", ".travis.yml", "ChangeLog.markdown", "Gemfile", "Guardfile", "LICENSE", "README.markdown", "Rakefile", "bin/homesick", "homesick.gemspec", "lib/homesick.rb", "lib/homesick/actions/file_actions.rb", "lib/homesick/actions/git_actions.rb", "lib/homesick/cli.rb", "lib/homesick/utils.rb", "lib/homesick/version.rb", "spec/homesick_cli_spec.rb", "spec/spec.opts", "spec/spec_helper.rb" ] s.homepage = "http://github.com/technicalpickles/homesick" s.licenses = ["MIT"] s.rubygems_version = "2.2.2" s.summary = "Your home directory is your castle. Don't leave your dotfiles behind." if s.respond_to? :specification_version then s.specification_version = 4 if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then s.add_runtime_dependency(%q, [">= 0.14.0"]) s.add_development_dependency(%q, [">= 0.8.7"]) s.add_development_dependency(%q, ["~> 3.1.0"]) s.add_development_dependency(%q, [">= 0"]) s.add_development_dependency(%q, [">= 0"]) s.add_development_dependency(%q, ["~> 0.5.0"]) s.add_development_dependency(%q, [">= 1.6.2"]) s.add_development_dependency(%q, [">= 0"]) s.add_development_dependency(%q, [">= 0"]) s.add_development_dependency(%q, ["~> 1.0.0"]) s.add_development_dependency(%q, [">= 0"]) else s.add_dependency(%q, [">= 0.14.0"]) s.add_dependency(%q, [">= 0.8.7"]) s.add_dependency(%q, ["~> 3.1.0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, ["~> 0.5.0"]) s.add_dependency(%q, [">= 1.6.2"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, ["~> 1.0.0"]) s.add_dependency(%q, [">= 0"]) end else s.add_dependency(%q, [">= 0.14.0"]) s.add_dependency(%q, [">= 0.8.7"]) s.add_dependency(%q, ["~> 3.1.0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, ["~> 0.5.0"]) s.add_dependency(%q, [">= 1.6.2"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, [">= 0"]) s.add_dependency(%q, ["~> 1.0.0"]) s.add_dependency(%q, [">= 0"]) end end homesick-1.1.3/.rspec0000644000004100000410000000001012642331233014451 0ustar www-datawww-data--color homesick-1.1.3/spec/0000755000004100000410000000000012642331233014277 5ustar www-datawww-datahomesick-1.1.3/spec/homesick_cli_spec.rb0000644000004100000410000007043212642331233020275 0ustar www-datawww-data# -*- encoding : utf-8 -*- require 'spec_helper' require 'capture-output' require 'pathname' describe Homesick::CLI do let(:home) { create_construct } after { home.destroy! } let(:castles) { home.directory('.homesick/repos') } let(:homesick) { Homesick::CLI.new } before { allow(homesick).to receive(:repos_dir).and_return(castles) } describe 'smoke tests' do context 'when running bin/homesick' do before do bin_path = Pathname.new(__FILE__).parent.parent @output = `#{bin_path.expand_path}/bin/homesick` end it 'should output some text when bin/homesick is called' do expect(@output.length).to be > 0 end end context 'when a git version that doesn\'t meet the minimum required is installed' do before do expect_any_instance_of(Homesick::Actions::GitActions).to receive(:`).and_return('git version 1.7.6') end it 'should raise an exception' do output = Capture.stdout { expect { Homesick::CLI.new }.to raise_error SystemExit } expect(output.chomp).to include(Homesick::Actions::GitActions::STRING) end end context 'when a git version that is the same as the minimum required is installed' do before do expect_any_instance_of(Homesick::Actions::GitActions).to receive(:`).at_least(:once).and_return("git version #{Homesick::Actions::GitActions::STRING}") end it 'should not raise an exception' do output = Capture.stdout { expect { Homesick::CLI.new }.not_to raise_error } expect(output.chomp).not_to include(Homesick::Actions::GitActions::STRING) end end context 'when a git version that is greater than the minimum required is installed' do before do expect_any_instance_of(Homesick::Actions::GitActions).to receive(:`).at_least(:once).and_return('git version 3.9.8') end it 'should not raise an exception' do output = Capture.stdout { expect { Homesick::CLI.new }.not_to raise_error } expect(output.chomp).not_to include(Homesick::Actions::GitActions::STRING) end end end describe 'clone' do context 'has a .homesickrc' do it 'runs the .homesickrc' do somewhere = create_construct local_repo = somewhere.directory('some_repo') local_repo.file('.homesickrc') do |file| file << "File.open(Dir.pwd + '/testing', 'w') do |f| f.print 'testing' end" end expect_any_instance_of(Thor::Shell::Basic).to receive(:yes?).with(be_a(String)).and_return(true) expect(homesick).to receive(:say_status).with('eval', kind_of(Pathname)) homesick.clone local_repo expect(castles.join('some_repo').join('testing')).to exist end end context 'of a file' do it 'symlinks existing directories' do somewhere = create_construct local_repo = somewhere.directory('wtf') homesick.clone local_repo expect(castles.join('wtf').readlink).to eq(local_repo) end context 'when it exists in a repo directory' do before do existing_castle = given_castle('existing_castle') @existing_dir = existing_castle.parent end it 'raises an error' do expect(homesick).not_to receive(:git_clone) expect { homesick.clone @existing_dir.to_s }.to raise_error(/already cloned/i) end end end it 'clones git repo like file:///path/to.git' do bare_repo = File.join(create_construct.to_s, 'dotfiles.git') system "git init --bare #{bare_repo} >/dev/null 2>&1" # Capture stderr to suppress message about cloning an empty repo. Capture.stderr do homesick.clone "file://#{bare_repo}" end expect(File.directory?(File.join(home.to_s, '.homesick/repos/dotfiles'))) .to be_truthy end it 'clones git repo like git://host/path/to.git' do expect(homesick).to receive(:git_clone) .with('git://github.com/technicalpickles/pickled-vim.git', destination: Pathname.new('pickled-vim')) homesick.clone 'git://github.com/technicalpickles/pickled-vim.git' end it 'clones git repo like git@host:path/to.git' do expect(homesick).to receive(:git_clone) .with('git@github.com:technicalpickles/pickled-vim.git', destination: Pathname.new('pickled-vim')) homesick.clone 'git@github.com:technicalpickles/pickled-vim.git' end it 'clones git repo like http://host/path/to.git' do expect(homesick).to receive(:git_clone) .with('http://github.com/technicalpickles/pickled-vim.git', destination: Pathname.new('pickled-vim')) homesick.clone 'http://github.com/technicalpickles/pickled-vim.git' end it 'clones git repo like http://host/path/to' do expect(homesick).to receive(:git_clone) .with('http://github.com/technicalpickles/pickled-vim', destination: Pathname.new('pickled-vim')) homesick.clone 'http://github.com/technicalpickles/pickled-vim' end it 'clones git repo like host-alias:repos.git' do expect(homesick).to receive(:git_clone).with('gitolite:pickled-vim.git', destination: Pathname.new('pickled-vim')) homesick.clone 'gitolite:pickled-vim.git' end it 'throws an exception when trying to clone a malformed uri like malformed' do expect(homesick).not_to receive(:git_clone) expect { homesick.clone 'malformed' }.to raise_error end it 'clones a github repo' do expect(homesick).to receive(:git_clone) .with('https://github.com/wfarr/dotfiles.git', destination: Pathname.new('dotfiles')) homesick.clone 'wfarr/dotfiles' end it 'accepts a destination', :focus do expect(homesick).to receive(:git_clone) .with('https://github.com/wfarr/dotfiles.git', destination: Pathname.new('other-name')) homesick.clone 'wfarr/dotfiles', 'other-name' end end describe 'rc' do let(:castle) { given_castle('glencairn') } context 'when told to do so' do before do expect_any_instance_of(Thor::Shell::Basic).to receive(:yes?).with(be_a(String)).and_return(true) end it 'executes the .homesickrc' do castle.file('.homesickrc') do |file| file << "File.open(Dir.pwd + '/testing', 'w') do |f| f.print 'testing' end" end expect(homesick).to receive(:say_status).with('eval', kind_of(Pathname)) homesick.rc castle expect(castle.join('testing')).to exist end end context 'when options[:force] == true' do let(:homesick) { Homesick::CLI.new [], force: true } before do expect_any_instance_of(Thor::Shell::Basic).to_not receive(:yes?) end it 'executes the .homesickrc' do castle.file('.homesickrc') do |file| file << "File.open(Dir.pwd + '/testing', 'w') do |f| f.print 'testing' end" end expect(homesick).to receive(:say_status).with('eval', kind_of(Pathname)) homesick.rc castle expect(castle.join('testing')).to exist end end context 'when told not to do so' do before do expect_any_instance_of(Thor::Shell::Basic).to receive(:yes?).with(be_a(String)).and_return(false) end it 'does not execute the .homesickrc' do castle.file('.homesickrc') do |file| file << "File.open(Dir.pwd + '/testing', 'w') do |f| f.print 'testing' end" end expect(homesick).to receive(:say_status).with('eval skip', /not evaling.+/, :blue) homesick.rc castle expect(castle.join('testing')).not_to exist end end end describe 'link_castle' do let(:castle) { given_castle('glencairn') } it 'links dotfiles from a castle to the home folder' do dotfile = castle.file('.some_dotfile') homesick.link('glencairn') expect(home.join('.some_dotfile').readlink).to eq(dotfile) end it 'links non-dotfiles from a castle to the home folder' do dotfile = castle.file('bin') homesick.link('glencairn') expect(home.join('bin').readlink).to eq(dotfile) end context 'when forced' do let(:homesick) { Homesick::CLI.new [], force: true } it 'can override symlinks to directories' do somewhere_else = create_construct existing_dotdir_link = home.join('.vim') FileUtils.ln_s somewhere_else, existing_dotdir_link dotdir = castle.directory('.vim') homesick.link('glencairn') expect(existing_dotdir_link.readlink).to eq(dotdir) end it 'can override existing directory' do existing_dotdir = home.directory('.vim') dotdir = castle.directory('.vim') homesick.link('glencairn') expect(existing_dotdir.readlink).to eq(dotdir) end end context "with '.config' in .homesick_subdir" do let(:castle) { given_castle('glencairn', ['.config']) } it 'can symlink in sub directory' do dotdir = castle.directory('.config') dotfile = dotdir.file('.some_dotfile') homesick.link('glencairn') home_dotdir = home.join('.config') expect(home_dotdir.symlink?).to eq(false) expect(home_dotdir.join('.some_dotfile').readlink).to eq(dotfile) end end context "with '.config/appA' in .homesick_subdir" do let(:castle) { given_castle('glencairn', ['.config/appA']) } it 'can symlink in nested sub directory' do dotdir = castle.directory('.config').directory('appA') dotfile = dotdir.file('.some_dotfile') homesick.link('glencairn') home_dotdir = home.join('.config').join('appA') expect(home_dotdir.symlink?).to eq(false) expect(home_dotdir.join('.some_dotfile').readlink).to eq(dotfile) end end context "with '.config' and '.config/someapp' in .homesick_subdir" do let(:castle) do given_castle('glencairn', ['.config', '.config/someapp']) end it 'can symlink under both of .config and .config/someapp' do config_dir = castle.directory('.config') config_dotfile = config_dir.file('.some_dotfile') someapp_dir = config_dir.directory('someapp') someapp_dotfile = someapp_dir.file('.some_appfile') homesick.link('glencairn') home_config_dir = home.join('.config') home_someapp_dir = home_config_dir.join('someapp') expect(home_config_dir.symlink?).to eq(false) expect(home_config_dir.join('.some_dotfile').readlink).to eq(config_dotfile) expect(home_someapp_dir.symlink?).to eq(false) expect(home_someapp_dir.join('.some_appfile').readlink).to eq(someapp_dotfile) end end context 'when call with no castle name' do let(:castle) { given_castle('dotfiles') } it 'using default castle name: "dotfiles"' do dotfile = castle.file('.some_dotfile') homesick.link expect(home.join('.some_dotfile').readlink).to eq(dotfile) end end end describe 'unlink' do let(:castle) { given_castle('glencairn') } it 'unlinks dotfiles in the home folder' do castle.file('.some_dotfile') homesick.link('glencairn') homesick.unlink('glencairn') expect(home.join('.some_dotfile')).not_to exist end it 'unlinks non-dotfiles from the home folder' do castle.file('bin') homesick.link('glencairn') homesick.unlink('glencairn') expect(home.join('bin')).not_to exist end context "with '.config' in .homesick_subdir" do let(:castle) { given_castle('glencairn', ['.config']) } it 'can unlink sub directories' do castle.directory('.config').file('.some_dotfile') homesick.link('glencairn') homesick.unlink('glencairn') home_dotdir = home.join('.config') expect(home_dotdir).to exist expect(home_dotdir.join('.some_dotfile')).not_to exist end end context "with '.config/appA' in .homesick_subdir" do let(:castle) { given_castle('glencairn', ['.config/appA']) } it 'can unsymlink in nested sub directory' do castle.directory('.config').directory('appA').file('.some_dotfile') homesick.link('glencairn') homesick.unlink('glencairn') home_dotdir = home.join('.config').join('appA') expect(home_dotdir).to exist expect(home_dotdir.join('.some_dotfile')).not_to exist end end context "with '.config' and '.config/someapp' in .homesick_subdir" do let(:castle) do given_castle('glencairn', ['.config', '.config/someapp']) end it 'can unsymlink under both of .config and .config/someapp' do config_dir = castle.directory('.config') config_dir.file('.some_dotfile') config_dir.directory('someapp').file('.some_appfile') homesick.link('glencairn') homesick.unlink('glencairn') home_config_dir = home.join('.config') home_someapp_dir = home_config_dir.join('someapp') expect(home_config_dir).to exist expect(home_config_dir.join('.some_dotfile')).not_to exist expect(home_someapp_dir).to exist expect(home_someapp_dir.join('.some_appfile')).not_to exist end end context 'when call with no castle name' do let(:castle) { given_castle('dotfiles') } it 'using default castle name: "dotfiles"' do castle.file('.some_dotfile') homesick.link homesick.unlink expect(home.join('.some_dotfile')).not_to exist end end end describe 'list' do it 'says each castle in the castle directory' do given_castle('zomg') given_castle('wtf/zomg') expect(homesick).to receive(:say_status) .with('zomg', 'git://github.com/technicalpickles/zomg.git', :cyan) expect(homesick).to receive(:say_status) .with('wtf/zomg', 'git://github.com/technicalpickles/zomg.git', :cyan) homesick.list end end describe 'status' do it 'says "nothing to commit" when there are no changes' do given_castle('castle_repo') text = Capture.stdout { homesick.status('castle_repo') } expect(text).to match(%r{nothing to commit \(create/copy files and use "git add" to track\)$}) end it 'says "Changes to be committed" when there are changes' do given_castle('castle_repo') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'castle_repo') text = Capture.stdout { homesick.status('castle_repo') } expect(text).to match(%r{Changes to be committed:.*new file:\s*home\/.some_rc_file}m) end end describe 'diff' do it 'outputs an empty message when there are no changes to commit' do given_castle('castle_repo') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'castle_repo') Capture.stdout do homesick.commit 'castle_repo', 'Adding a file to the test' end text = Capture.stdout { homesick.diff('castle_repo') } expect(text).to eq('') end it 'outputs a diff message when there are changes to commit' do given_castle('castle_repo') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'castle_repo') Capture.stdout do homesick.commit 'castle_repo', 'Adding a file to the test' end File.open(some_rc_file.to_s, 'w') do |file| file.puts 'Some test text' end text = Capture.stdout { homesick.diff('castle_repo') } expect(text).to match(/diff --git.+Some test text$/m) end end describe 'show_path' do it 'says the path of a castle' do castle = given_castle('castle_repo') expect(homesick).to receive(:say).with(castle.dirname) homesick.show_path('castle_repo') end end describe 'pull' do it 'performs a pull, submodule init and update when the given castle exists' do given_castle('castle_repo') allow(homesick).to receive(:system).once.with('git pull --quiet') allow(homesick).to receive(:system).once.with('git submodule --quiet init') allow(homesick).to receive(:system).once.with('git submodule --quiet update --init --recursive >/dev/null 2>&1') homesick.pull 'castle_repo' end it 'prints an error message when trying to pull a non-existant castle' do expect(homesick).to receive('say_status').once .with(:error, /Could not pull castle_repo, expected .* exist and contain dotfiles/, :red) expect { homesick.pull 'castle_repo' }.to raise_error(SystemExit) end describe '--all' do it 'pulls each castle when invoked with --all' do given_castle('castle_repo') given_castle('glencairn') allow(homesick).to receive(:system).exactly(2).times.with('git pull --quiet') allow(homesick).to receive(:system).exactly(2).times .with('git submodule --quiet init') allow(homesick).to receive(:system).exactly(2).times .with('git submodule --quiet update --init --recursive >/dev/null 2>&1') Capture.stdout do Capture.stderr { homesick.invoke 'pull', [], all: true } end end end end describe 'push' do it 'performs a git push on the given castle' do given_castle('castle_repo') allow(homesick).to receive(:system).once.with('git push') homesick.push 'castle_repo' end it 'prints an error message when trying to push a non-existant castle' do expect(homesick).to receive('say_status').once .with(:error, /Could not push castle_repo, expected .* exist and contain dotfiles/, :red) expect { homesick.push 'castle_repo' }.to raise_error(SystemExit) end end describe 'track' do it 'moves the tracked file into the castle' do castle = given_castle('castle_repo') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'castle_repo') tracked_file = castle.join('.some_rc_file') expect(tracked_file).to exist expect(some_rc_file.readlink).to eq(tracked_file) end it 'handles files with parens' do castle = given_castle('castle_repo') some_rc_file = home.file 'Default (Linux).sublime-keymap' homesick.track(some_rc_file.to_s, 'castle_repo') tracked_file = castle.join('Default (Linux).sublime-keymap') expect(tracked_file).to exist expect(some_rc_file.readlink).to eq(tracked_file) end it 'tracks a file in nested folder structure' do castle = given_castle('castle_repo') some_nested_file = home.file('some/nested/file.txt') homesick.track(some_nested_file.to_s, 'castle_repo') tracked_file = castle.join('some/nested/file.txt') expect(tracked_file).to exist expect(some_nested_file.readlink).to eq(tracked_file) end it 'tracks a nested directory' do castle = given_castle('castle_repo') some_nested_dir = home.directory('some/nested/directory/') homesick.track(some_nested_dir.to_s, 'castle_repo') tracked_file = castle.join('some/nested/directory/') expect(tracked_file).to exist expect(some_nested_dir.realpath).to eq(tracked_file.realpath) end context 'when call with no castle name' do it 'using default castle name: "dotfiles"' do castle = given_castle('dotfiles') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s) tracked_file = castle.join('.some_rc_file') expect(tracked_file).to exist expect(some_rc_file.readlink).to eq(tracked_file) end end describe 'commit' do it 'has a commit message when the commit succeeds' do given_castle('castle_repo') some_rc_file = home.file '.a_random_rc_file' homesick.track(some_rc_file.to_s, 'castle_repo') text = Capture.stdout do homesick.commit('castle_repo', 'Test message') end expect(text).to match(/^\[master \(root-commit\) \w+\] Test message/) end end # Note that this is a test for the subdir_file related feature of track, # not for the subdir_file method itself. describe 'subdir_file' do it 'adds the nested files parent to the subdir_file' do castle = given_castle('castle_repo') some_nested_file = home.file('some/nested/file.txt') homesick.track(some_nested_file.to_s, 'castle_repo') subdir_file = castle.parent.join(Homesick::SUBDIR_FILENAME) File.open(subdir_file, 'r') do |f| expect(f.readline).to eq("some/nested\n") end end it 'does NOT add anything if the files parent is already listed' do castle = given_castle('castle_repo') some_nested_file = home.file('some/nested/file.txt') other_nested_file = home.file('some/nested/other.txt') homesick.track(some_nested_file.to_s, 'castle_repo') homesick.track(other_nested_file.to_s, 'castle_repo') subdir_file = castle.parent.join(Homesick::SUBDIR_FILENAME) File.open(subdir_file, 'r') do |f| expect(f.readlines.size).to eq(1) end end it 'removes the parent of a tracked file from the subdir_file if the parent itself is tracked' do castle = given_castle('castle_repo') some_nested_file = home.file('some/nested/file.txt') nested_parent = home.directory('some/nested/') homesick.track(some_nested_file.to_s, 'castle_repo') homesick.track(nested_parent.to_s, 'castle_repo') subdir_file = castle.parent.join(Homesick::SUBDIR_FILENAME) File.open(subdir_file, 'r') do |f| f.each_line { |line| expect(line).not_to eq("some/nested\n") } end end end end describe 'destroy' do it 'removes the symlink files' do expect_any_instance_of(Thor::Shell::Basic).to receive(:yes?).and_return('y') given_castle('stronghold') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'stronghold') homesick.destroy('stronghold') expect(some_rc_file).not_to be_exist end it 'deletes the cloned repository' do expect_any_instance_of(Thor::Shell::Basic).to receive(:yes?).and_return('y') castle = given_castle('stronghold') some_rc_file = home.file '.some_rc_file' homesick.track(some_rc_file.to_s, 'stronghold') homesick.destroy('stronghold') expect(castle).not_to be_exist end end describe 'cd' do it "cd's to the root directory of the given castle" do given_castle('castle_repo') expect(homesick).to receive('inside').once.with(kind_of(Pathname)).and_yield expect(homesick).to receive('system').once.with(ENV['SHELL']) Capture.stdout { homesick.cd 'castle_repo' } end it 'returns an error message when the given castle does not exist' do expect(homesick).to receive('say_status').once .with(:error, /Could not cd castle_repo, expected .* exist and contain dotfiles/, :red) expect { homesick.cd 'castle_repo' }.to raise_error(SystemExit) end end describe 'open' do it 'opens the system default editor in the root of the given castle' do # Make sure calls to ENV use default values for most things... allow(ENV).to receive(:[]).and_call_original # Set a default value for 'EDITOR' just in case none is set allow(ENV).to receive(:[]).with('EDITOR').and_return('vim') given_castle 'castle_repo' expect(homesick).to receive('inside').once.with(kind_of(Pathname)).and_yield expect(homesick).to receive('system').once.with('vim .') Capture.stdout { homesick.open 'castle_repo' } end it 'returns an error message when the $EDITOR environment variable is not set' do # Set the default editor to make sure it fails. allow(ENV).to receive(:[]).with('EDITOR').and_return(nil) expect(homesick).to receive('say_status').once .with(:error, 'The $EDITOR environment variable must be set to use this command', :red) expect { homesick.open 'castle_repo' }.to raise_error(SystemExit) end it 'returns an error message when the given castle does not exist' do # Set a default just in case none is set allow(ENV).to receive(:[]).with('EDITOR').and_return('vim') allow(homesick).to receive('say_status').once .with(:error, /Could not open castle_repo, expected .* exist and contain dotfiles/, :red) expect { homesick.open 'castle_repo' }.to raise_error(SystemExit) end end describe 'version' do it 'prints the current version of homesick' do text = Capture.stdout { homesick.version } expect(text.chomp).to match(/#{Regexp.escape(Homesick::Version::STRING)}/) end end describe 'exec' do before do given_castle 'castle_repo' end it 'executes a single command with no arguments inside a given castle' do allow(homesick).to receive('inside').once.with(kind_of(Pathname)).and_yield allow(homesick).to receive('say_status').once .with(be_a(String), be_a(String), :green) allow(homesick).to receive('system').once.with('ls') Capture.stdout { homesick.exec 'castle_repo', 'ls' } end it 'executes a single command with arguments inside a given castle' do allow(homesick).to receive('inside').once.with(kind_of(Pathname)).and_yield allow(homesick).to receive('say_status').once .with(be_a(String), be_a(String), :green) allow(homesick).to receive('system').once.with('ls -la') Capture.stdout { homesick.exec 'castle_repo', 'ls', '-la' } end it 'raises an error when the method is called without a command' do allow(homesick).to receive('say_status').once .with(:error, be_a(String), :red) allow(homesick).to receive('exit').once.with(1) Capture.stdout { homesick.exec 'castle_repo' } end context 'pretend' do it 'does not execute a command when the pretend option is passed' do allow(homesick).to receive('say_status').once .with(be_a(String), match(/.*Would execute.*/), :green) expect(homesick).to receive('system').never Capture.stdout { homesick.invoke 'exec', %w(castle_repo ls -la), pretend: true } end end context 'quiet' do it 'does not print status information when quiet is passed' do expect(homesick).to receive('say_status').never allow(homesick).to receive('system').once .with('ls -la') Capture.stdout { homesick.invoke 'exec', %w(castle_repo ls -la), quiet: true } end end end describe 'exec_all' do before do given_castle 'castle_repo' given_castle 'another_castle_repo' end it 'executes a command without arguments inside the root of each cloned castle' do allow(homesick).to receive('inside_each_castle').exactly(:twice).and_yield('castle_repo').and_yield('another_castle_repo') allow(homesick).to receive('say_status').at_least(:once) .with(be_a(String), be_a(String), :green) allow(homesick).to receive('system').at_least(:once).with('ls') Capture.stdout { homesick.exec_all 'ls' } end it 'executes a command with arguments inside the root of each cloned castle' do allow(homesick).to receive('inside_each_castle').exactly(:twice).and_yield('castle_repo').and_yield('another_castle_repo') allow(homesick).to receive('say_status').at_least(:once) .with(be_a(String), be_a(String), :green) allow(homesick).to receive('system').at_least(:once).with('ls -la') Capture.stdout { homesick.exec_all 'ls', '-la' } end it 'raises an error when the method is called without a command' do allow(homesick).to receive('say_status').once .with(:error, be_a(String), :red) allow(homesick).to receive('exit').once.with(1) Capture.stdout { homesick.exec_all } end context 'pretend' do it 'does not execute a command when the pretend option is passed' do allow(homesick).to receive('say_status').at_least(:once) .with(be_a(String), match(/.*Would execute.*/), :green) expect(homesick).to receive('system').never Capture.stdout { homesick.invoke 'exec_all', %w(ls -la), pretend: true } end end context 'quiet' do it 'does not print status information when quiet is passed' do expect(homesick).to receive('say_status').never allow(homesick).to receive('system').at_least(:once) .with('ls -la') Capture.stdout { homesick.invoke 'exec_all', %w(ls -la), quiet: true } end end end end homesick-1.1.3/spec/spec.opts0000644000004100000410000000001012642331233016127 0ustar www-datawww-data--color homesick-1.1.3/spec/spec_helper.rb0000644000004100000410000000223012642331233017112 0ustar www-datawww-datarequire 'coveralls' Coveralls.wear! $LOAD_PATH.unshift(File.dirname(__FILE__)) $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib')) require 'homesick' require 'rspec' require 'test_construct' require 'tempfile' RSpec.configure do |config| config.include TestConstruct::Helpers config.expect_with(:rspec) { |c| c.syntax = :expect } config.before { ENV['HOME'] = home.to_s } config.before { silence! } def silence! allow(homesick).to receive(:say_status) end def given_castle(path, subdirs = []) name = Pathname.new(path).basename castles.directory(path) do |castle| Dir.chdir(castle) do system 'git init >/dev/null 2>&1' system 'git config user.email "test@test.com"' system 'git config user.name "Test Name"' system "git remote add origin git://github.com/technicalpickles/#{name}.git >/dev/null 2>&1" if subdirs subdir_file = castle.join(Homesick::SUBDIR_FILENAME) subdirs.each do |subdir| File.open(subdir_file, 'a') { |file| file.write "\n#{subdir}\n" } end end return castle.directory('home') end end end end homesick-1.1.3/.travis.yml0000644000004100000410000000007612642331233015461 0ustar www-datawww-datalanguage: ruby rvm: - 2.1.0 - 2.0.0 - 1.9.3 sudo: false homesick-1.1.3/lib/0000755000004100000410000000000012642331233014113 5ustar www-datawww-datahomesick-1.1.3/lib/homesick.rb0000644000004100000410000000057012642331233016244 0ustar www-datawww-data# -*- encoding : utf-8 -*- require 'homesick/actions/file_actions' require 'homesick/actions/git_actions' require 'homesick/version' require 'homesick/utils' require 'homesick/cli' # Homesick's top-level module module Homesick GITHUB_NAME_REPO_PATTERN = %r{\A([A-Za-z0-9_-]+/[A-Za-z0-9_-]+)\Z} SUBDIR_FILENAME = '.homesick_subdir' DEFAULT_CASTLE_NAME = 'dotfiles' end homesick-1.1.3/lib/homesick/0000755000004100000410000000000012642331233015715 5ustar www-datawww-datahomesick-1.1.3/lib/homesick/utils.rb0000644000004100000410000001314312642331233017404 0ustar www-datawww-data# -*- encoding : utf-8 -*- require 'pathname' module Homesick # Various utility methods that are used by Homesick module Utils QUIETABLE = ['say_status'] PRETENDABLE = ['system'] QUIETABLE.each do |method_name| define_method(method_name) do |*args| super(*args) unless options[:quiet] end end PRETENDABLE.each do |method_name| define_method(method_name) do |*args| super(*args) unless options[:pretend] end end protected def home_dir @home_dir ||= Pathname.new(ENV['HOME'] || '~').realpath end def repos_dir @repos_dir ||= home_dir.join('.homesick', 'repos').expand_path end def castle_dir(name) repos_dir.join(name, 'home') end def check_castle_existance(name, action) return if castle_dir(name).exist? say_status :error, "Could not #{action} #{name}, expected #{castle_dir(name)} exist and contain dotfiles", :red exit(1) end def all_castles dirs = Pathname.glob("#{repos_dir}/**/.git", File::FNM_DOTMATCH) # reject paths that lie inside another castle, like git submodules dirs.reject do |dir| dirs.any? do |other| dir != other && dir.fnmatch(other.parent.join('*').to_s) end end end def inside_each_castle all_castles.each do |git_dir| castle = git_dir.dirname Dir.chdir castle do # so we can call git config from the right contxt yield castle end end end def update_castle(castle) check_castle_existance(castle, 'pull') inside repos_dir.join(castle) do git_pull git_submodule_init git_submodule_update end end def commit_castle(castle, message) check_castle_existance(castle, 'commit') inside repos_dir.join(castle) do git_commit_all message: message end end def push_castle(castle) check_castle_existance(castle, 'push') inside repos_dir.join(castle) do git_push end end def subdir_file(castle) repos_dir.join(castle, SUBDIR_FILENAME) end def subdirs(castle) subdir_filepath = subdir_file(castle) subdirs = [] if subdir_filepath.exist? subdir_filepath.readlines.each do |subdir| subdirs.push(subdir.chomp) end end subdirs end def subdir_add(castle, path) subdir_filepath = subdir_file(castle) File.open(subdir_filepath, 'a+') do |subdir| subdir.puts path unless subdir.readlines.reduce(false) do |memo, line| line.eql?("#{path}\n") || memo end end inside castle_dir(castle) do git_add subdir_filepath end end def subdir_remove(castle, path) subdir_filepath = subdir_file(castle) if subdir_filepath.exist? lines = IO.readlines(subdir_filepath).delete_if do |line| line == "#{path}\n" end File.open(subdir_filepath, 'w') { |manfile| manfile.puts lines } end inside castle_dir(castle) do git_add subdir_filepath end end def move_dir_contents(target, dir_path) child_files = dir_path.children child_files.each do |child| target_path = target.join(child.basename) if target_path.exist? if more_recent?(child, target_path) && target.file? target_path.delete mv child, target end next end mv child, target end end def more_recent?(first, second) first_p = Pathname.new(first) second_p = Pathname.new(second) first_p.mtime > second_p.mtime && !first_p.symlink? end def collision_accepted?(destination, source) fail "Arguments must be instances of Pathname, #{destination.class.name} and #{source.class.name} given" unless destination.instance_of?(Pathname) && source.instance_of?(Pathname) options[:force] || shell.file_collision(destination) { source } end def each_file(castle, basedir, subdirs) absolute_basedir = Pathname.new(basedir).expand_path inside basedir do files = Pathname.glob('{.*,*}').reject do |a| ['.', '..'].include?(a.to_s) end files.each do |path| absolute_path = path.expand_path castle_home = castle_dir(castle) # make ignore dirs ignore_dirs = [] subdirs.each do |subdir| # ignore all parent of each line in subdir file Pathname.new(subdir).ascend do |p| ignore_dirs.push(p) end end # ignore dirs written in subdir file matched = false ignore_dirs.uniq.each do |ignore_dir| if absolute_path == castle_home.join(ignore_dir) matched = true break end end next if matched relative_dir = absolute_basedir.relative_path_from(castle_home) home_path = home_dir.join(relative_dir).join(path) yield(absolute_path, home_path) end end end def unsymlink_each(castle, basedir, subdirs) each_file(castle, basedir, subdirs) do |_absolute_path, home_path| rm_link home_path end end def symlink_each(castle, basedir, subdirs) each_file(castle, basedir, subdirs) do |absolute_path, home_path| ln_s absolute_path, home_path end end def setup_castle(path) if path.join('.gitmodules').exist? inside path do git_submodule_init git_submodule_update end end rc(path) end end end homesick-1.1.3/lib/homesick/actions/0000755000004100000410000000000012642331233017355 5ustar www-datawww-datahomesick-1.1.3/lib/homesick/actions/file_actions.rb0000644000004100000410000000543412642331233022347 0ustar www-datawww-data# -*- encoding : utf-8 -*- module Homesick module Actions # File-related helper methods for Homesick module FileActions def mv(source, destination) source = Pathname.new(source) destination = Pathname.new(destination + source.basename) case when destination.exist? && (options[:force] || shell.file_collision(destination) { source }) say_status :conflict, "#{destination} exists", :red FileUtils.mv source, destination unless options[:pretend] else FileUtils.mv source, destination unless options[:pretend] end end def rm_rf(dir) say_status "rm -rf #{dir}", '', :green FileUtils.rm_r dir, force: true end def rm_link(target) target = Pathname.new(target) if target.symlink? say_status :unlink, "#{target.expand_path}", :green FileUtils.rm_rf target else say_status :conflict, "#{target} is not a symlink", :red end end def rm(file) say_status "rm #{file}", '', :green FileUtils.rm file, force: true end def rm_r(dir) say_status "rm -r #{dir}", '', :green FileUtils.rm_r dir end def ln_s(source, destination) source = Pathname.new(source) destination = Pathname.new(destination) FileUtils.mkdir_p destination.dirname action = if destination.symlink? && destination.readlink == source :identical elsif destination.symlink? :symlink_conflict elsif destination.exist? :conflict else :success end handle_symlink_action action, source, destination end def handle_symlink_action(action, source, destination) case action when :identical say_status :identical, destination.expand_path, :blue when :symlink_conflict say_status :conflict, "#{destination} exists and points to #{destination.readlink}", :red FileUtils.rm destination FileUtils.ln_s source, destination, force: true unless options[:pretend] when :conflict say_status :conflict, "#{destination} exists", :red if collision_accepted?(destination, source) FileUtils.rm_r destination, force: true unless options[:pretend] FileUtils.ln_s source, destination, force: true unless options[:pretend] end else say_status :symlink, "#{source.expand_path} to #{destination.expand_path}", :green FileUtils.ln_s source, destination unless options[:pretend] end end end end end homesick-1.1.3/lib/homesick/actions/git_actions.rb0000644000004100000410000000664212642331233022215 0ustar www-datawww-data# -*- encoding : utf-8 -*- module Homesick module Actions # Git-related helper methods for Homesick module GitActions # Information on the minimum git version required for Homesick MIN_VERSION = { major: 1, minor: 8, patch: 0 } STRING = MIN_VERSION.values.join('.') def git_version_correct? info = `git --version`.scan(/(\d+)\.(\d+)\.(\d+)/).flatten.map(&:to_i) return false unless info.count == 3 current_version = Hash[[:major, :minor, :patch].zip(info)] return true if current_version.eql?(MIN_VERSION) return true if current_version[:major] > MIN_VERSION[:major] return true if current_version[:major] == MIN_VERSION[:major] && current_version[:minor] > MIN_VERSION[:minor] return true if current_version[:major] == MIN_VERSION[:major] && current_version[:minor] == MIN_VERSION[:minor] && current_version[:patch] >= MIN_VERSION[:patch] false end # TODO: move this to be more like thor's template, empty_directory, etc def git_clone(repo, config = {}) config ||= {} destination = config[:destination] || File.basename(repo, '.git') destination = Pathname.new(destination) unless destination.is_a?(Pathname) FileUtils.mkdir_p destination.dirname if destination.directory? say_status :exist, destination.expand_path, :blue else say_status 'git clone', "#{repo} to #{destination.expand_path}", :green system "git clone -q --config push.default=upstream --recursive #{repo} #{destination}" end end def git_init(path = '.') path = Pathname.new(path) inside path do if path.join('.git').exist? say_status 'git init', 'already initialized', :blue else say_status 'git init', '' system 'git init >/dev/null' end end end def git_remote_add(name, url) existing_remote = `git config remote.#{name}.url`.chomp existing_remote = nil if existing_remote == '' if existing_remote say_status 'git remote', "#{name} already exists", :blue else say_status 'git remote', "add #{name} #{url}" system "git remote add #{name} #{url}" end end def git_submodule_init say_status 'git submodule', 'init', :green system 'git submodule --quiet init' end def git_submodule_update say_status 'git submodule', 'update', :green system 'git submodule --quiet update --init --recursive >/dev/null 2>&1' end def git_pull say_status 'git pull', '', :green system 'git pull --quiet' end def git_push say_status 'git push', '', :green system 'git push' end def git_commit_all(config = {}) say_status 'git commit all', '', :green if config[:message] system %(git commit -a -m "#{config[:message]}") else system 'git commit -v -a' end end def git_add(file) say_status 'git add file', '', :green system "git add '#{file}'" end def git_status say_status 'git status', '', :green system 'git status' end def git_diff say_status 'git diff', '', :green system 'git diff' end end end end homesick-1.1.3/lib/homesick/cli.rb0000644000004100000410000002551212642331233017016 0ustar www-datawww-data# -*- encoding : utf-8 -*- require 'thor' module Homesick # Homesick's command line interface class CLI < Thor include Thor::Actions include Homesick::Actions::FileActions include Homesick::Actions::GitActions include Homesick::Version include Homesick::Utils add_runtime_options! map '-v' => :version map '--version' => :version # Retain a mapped version of the symlink command for compatibility. map symlink: :link def initialize(args = [], options = {}, config = {}) super # Check if git is installed unless git_version_correct? say_status :error, "Git version >= #{Homesick::Actions::GitActions::STRING} must be installed to use Homesick", :red exit(1) end # Hack in support for diffing symlinks # Also adds support for checking if destination or content is a directory shell_metaclass = class << shell; self; end shell_metaclass.send(:define_method, :show_diff) do |destination, content| destination = Pathname.new(destination) content = Pathname.new(content) return 'Unable to create diff: destination or content is a directory' if destination.directory? || content.directory? return super(destination, content) unless destination.symlink? say "- #{destination.readlink}", :red, true say "+ #{content.expand_path}", :green, true end end desc 'clone URI CASTLE_NAME', 'Clone +uri+ as a castle with name CASTLE_NAME for homesick' def clone(uri, destination=nil) destination = Pathname.new(destination) unless destination.nil? inside repos_dir do if File.exist?(uri) uri = Pathname.new(uri).expand_path fail "Castle already cloned to #{uri}" if uri.to_s.start_with?(repos_dir.to_s) destination = uri.basename if destination.nil? ln_s uri, destination elsif uri =~ GITHUB_NAME_REPO_PATTERN destination = Pathname.new(uri).basename if destination.nil? git_clone "https://github.com/#{Regexp.last_match[1]}.git", destination: destination elsif uri =~ /%r([^%r]*?)(\.git)?\Z/ || uri =~ /[^:]+:([^:]+)(\.git)?\Z/ destination = Pathname.new(Regexp.last_match[1].gsub(/\.git$/, '')).basename if destination.nil? git_clone uri, destination: destination else fail "Unknown URI format: #{uri}" end setup_castle(destination) end end desc 'rc CASTLE', 'Run the .homesickrc for the specified castle' method_option :force, default: false, desc: 'Evaluate .homesickrc without prompting.' def rc(name = DEFAULT_CASTLE_NAME) inside repos_dir do destination = Pathname.new(name) homesickrc = destination.join('.homesickrc').expand_path return unless homesickrc.exist? proceed = options[:force] || shell.yes?("#{name} has a .homesickrc. Proceed with evaling it? (This could be destructive)") return say_status 'eval skip', "not evaling #{homesickrc}, #{destination} may need manual configuration", :blue unless proceed say_status 'eval', homesickrc inside destination do eval homesickrc.read, binding, homesickrc.expand_path.to_s end end end desc 'pull CASTLE', 'Update the specified castle' method_option :all, type: :boolean, default: false, required: false, desc: 'Update all cloned castles' def pull(name = DEFAULT_CASTLE_NAME) if options[:all] inside_each_castle do |castle| say castle.to_s.gsub(repos_dir.to_s + '/', '') + ':' update_castle castle end else update_castle name end end desc 'commit CASTLE MESSAGE', "Commit the specified castle's changes" def commit(name = DEFAULT_CASTLE_NAME, message = nil) commit_castle name, message end desc 'push CASTLE', 'Push the specified castle' def push(name = DEFAULT_CASTLE_NAME) push_castle name end desc 'unlink CASTLE', 'Unsymlinks all dotfiles from the specified castle' def unlink(name = DEFAULT_CASTLE_NAME) check_castle_existance(name, 'symlink') inside castle_dir(name) do subdirs = subdirs(name) # unlink files unsymlink_each(name, castle_dir(name), subdirs) # unlink files in subdirs subdirs.each do |subdir| unsymlink_each(name, subdir, subdirs) end end end desc 'link CASTLE', 'Symlinks all dotfiles from the specified castle' method_option :force, default: false, desc: 'Overwrite existing conflicting symlinks without prompting.' def link(name = DEFAULT_CASTLE_NAME) check_castle_existance(name, 'symlink') inside castle_dir(name) do subdirs = subdirs(name) # link files symlink_each(name, castle_dir(name), subdirs) # link files in subdirs subdirs.each do |subdir| symlink_each(name, subdir, subdirs) end end end desc 'track FILE CASTLE', 'add a file to a castle' def track(file, castle = DEFAULT_CASTLE_NAME) castle = Pathname.new(castle) file = Pathname.new(file.chomp('/')) check_castle_existance(castle, 'track') absolute_path = file.expand_path relative_dir = absolute_path.relative_path_from(home_dir).dirname castle_path = Pathname.new(castle_dir(castle)).join(relative_dir) FileUtils.mkdir_p castle_path # Are we already tracking this or anything inside it? target = Pathname.new(castle_path.join(file.basename)) if target.exist? if absolute_path.directory? move_dir_contents(target, absolute_path) absolute_path.rmtree subdir_remove(castle, relative_dir + file.basename) elsif more_recent? absolute_path, target target.delete mv absolute_path, castle_path else say_status(:track, "#{target} already exists, and is more recent than #{file}. Run 'homesick SYMLINK CASTLE' to create symlinks.", :blue) end else mv absolute_path, castle_path end inside home_dir do absolute_path = castle_path + file.basename home_path = home_dir + relative_dir + file.basename ln_s absolute_path, home_path end inside castle_path do git_add absolute_path end # are we tracking something nested? Add the parent dir to the manifest subdir_add(castle, relative_dir) unless relative_dir.eql?(Pathname.new('.')) end desc 'list', 'List cloned castles' def list inside_each_castle do |castle| say_status castle.relative_path_from(repos_dir).to_s, `git config remote.origin.url`.chomp, :cyan end end desc 'status CASTLE', 'Shows the git status of a castle' def status(castle = DEFAULT_CASTLE_NAME) check_castle_existance(castle, 'status') inside repos_dir.join(castle) do git_status end end desc 'diff CASTLE', 'Shows the git diff of uncommitted changes in a castle' def diff(castle = DEFAULT_CASTLE_NAME) check_castle_existance(castle, 'diff') inside repos_dir.join(castle) do git_diff end end desc 'show_path CASTLE', 'Prints the path of a castle' def show_path(castle = DEFAULT_CASTLE_NAME) check_castle_existance(castle, 'show_path') say repos_dir.join(castle) end desc 'generate PATH', 'generate a homesick-ready git repo at PATH' def generate(castle) castle = Pathname.new(castle).expand_path github_user = `git config github.user`.chomp github_user = nil if github_user == '' github_repo = castle.basename empty_directory castle inside castle do git_init if github_user url = "git@github.com:#{github_user}/#{github_repo}.git" git_remote_add 'origin', url end empty_directory 'home' end end desc 'destroy CASTLE', 'Delete all symlinks and remove the cloned repository' def destroy(name) check_castle_existance name, 'destroy' return unless shell.yes?('This will destroy your castle irreversible! Are you sure?') unlink(name) rm_rf repos_dir.join(name) end desc 'cd CASTLE', 'Open a new shell in the root of the given castle' def cd(castle = DEFAULT_CASTLE_NAME) check_castle_existance castle, 'cd' castle_dir = repos_dir.join(castle) say_status "cd #{castle_dir.realpath}", "Opening a new shell in castle '#{castle}'. To return to the original one exit from the new shell.", :green inside castle_dir do system(ENV['SHELL']) end end desc 'open CASTLE', 'Open your default editor in the root of the given castle' def open(castle = DEFAULT_CASTLE_NAME) unless ENV['EDITOR'] say_status :error, 'The $EDITOR environment variable must be set to use this command', :red exit(1) end check_castle_existance castle, 'open' castle_dir = repos_dir.join(castle) say_status "#{castle_dir.realpath}: #{ENV['EDITOR']} .", "Opening the root directory of castle '#{castle}' in editor '#{ENV['EDITOR']}'.", :green inside castle_dir do system("#{ENV['EDITOR']} .") end end desc 'exec CASTLE COMMAND', 'Execute a single shell command inside the root of a castle' def exec(castle, *args) check_castle_existance castle, 'exec' unless args.count > 0 say_status :error, 'You must pass a shell command to execute', :red exit(1) end full_command = args.join(' ') say_status "exec '#{full_command}'", "#{options[:pretend] ? 'Would execute' : 'Executing command'} '#{full_command}' in castle '#{castle}'", :green inside repos_dir.join(castle) do system(full_command) end end desc 'exec_all COMMAND', 'Execute a single shell command inside the root of every cloned castle' def exec_all(*args) unless args.count > 0 say_status :error, 'You must pass a shell command to execute', :red exit(1) end full_command = args.join(' ') inside_each_castle do |castle| say_status "exec '#{full_command}'", "#{options[:pretend] ? 'Would execute' : 'Executing command'} '#{full_command}' in castle '#{castle}'", :green system(full_command) end end desc 'version', 'Display the current version of homesick' def version say Homesick::Version::STRING end end end homesick-1.1.3/lib/homesick/version.rb0000644000004100000410000000043212642331233017726 0ustar www-datawww-data# -*- encoding : utf-8 -*- module Homesick # A representation of Homesick's version number in constants, including a # String of the entire version number module Version MAJOR = 1 MINOR = 1 PATCH = 3 STRING = [MAJOR, MINOR, PATCH].compact.join('.') end end homesick-1.1.3/README.markdown0000644000004100000410000001343212642331233016051 0ustar www-datawww-data# homesick [![Gem Version](https://badge.fury.io/rb/homesick.svg)](http://badge.fury.io/rb/homesick) [![Build Status](https://travis-ci.org/technicalpickles/homesick.svg?branch=master)](https://travis-ci.org/technicalpickles/homesick) [![Dependency Status](https://gemnasium.com/technicalpickles/homesick.svg)](https://gemnasium.com/technicalpickles/homesick) [![Coverage Status](https://coveralls.io/repos/technicalpickles/homesick/badge.png)](https://coveralls.io/r/technicalpickles/homesick) [![Code Climate](https://codeclimate.com/github/technicalpickles/homesick.svg)](https://codeclimate.com/github/technicalpickles/homesick) [![Gitter chat](https://badges.gitter.im/technicalpickles/homesick.svg)](https://gitter.im/technicalpickles/homesick) Your home directory is your castle. Don't leave your dotfiles behind. Homesick is sorta like [rip](http://github.com/defunkt/rip), but for dotfiles. It uses git to clone a repository containing dotfiles, and saves them in `~/.homesick`. It then allows you to symlink all the dotfiles into place with a single command. We call a repository that is compatible with homesick to be a 'castle'. To act as a castle, a repository must be organized like so: * Contains a 'home' directory * 'home' contains any number of files and directories that begin with '.' To get started, install homesick first: gem install homesick Next, you use the homesick command to clone a castle: homesick clone git://github.com/technicalpickles/pickled-vim.git Alternatively, if it's on github, there's a slightly shorter way: homesick clone technicalpickles/pickled-vim With the castle cloned, you can now link its contents into your home dir: homesick symlink pickled-vim You can remove symlinks anytime when you don't need them anymore homesick unlink pickled-vim If you need to add further configuration steps you can add these in a file called '.homesickrc' in the root of a castle. Once you've cloned a castle with a .homesickrc run the configuration with: homesick rc CASTLE The contents of the .homesickrc file must be valid Ruby code as the file will be executed with Ruby's eval construct. The .homesickrc is also passed the current homesick object during its execution and this is available within the .homesickrc file as the 'self' variable. As the rc operation can be destructive the command normally asks for confirmation before proceeding. You can bypass this by passing the '--force' option, for example `homesick rc --force CASTLE`. If you're not sure what castles you have around, you can easily list them: homesick list To pull your castle (or all castles): homesick pull --all|CASTLE To commit your castle's changes: homesick commit CASTLE To push your castle: homesick push CASTLE To open a terminal in the root of a castle: homesick cd CASTLE To open your default editor in the root of a castle (the $EDITOR environment variable must be set): homesick open CASTLE To execute a shell command inside the root directory of a given castle: homesick exec CASTLE COMMAND To execute a shell command inside the root directory of every cloned castle: homesick exec_all COMMAND Not sure what else homesick has up its sleeve? There's always the built in help: homesick help If you ever want to see what version of homesick you have type: homesick version|-v|--version ## .homesick_subdir `homesick symlink` basically makes symlink to only first depth in `castle/home`. If you want to link nested files/directories, please use .homesick_subdir. For example, when you have castle like this: castle/home `-- .config `-- fooapp |-- config1 |-- config2 `-- config3 and have home like this: $ tree -a ~ |-- .config | `-- barapp | |-- config1 | |-- config2 | `-- config3 `-- .emacs.d |-- elisp `-- inits You may want to symlink only to `castle/home/.config/fooapp` instead of `castle/home/.config` because you already have `~/.config/barapp`. In this case, you can use .homesick_subdir. Please write "directories you want to look up sub directories (instead of just first depth)" in this file. castle/.homesick_subdir .config and run `homesick symlink CASTLE`. The result is: ~ |-- .config | |-- barapp | | |-- config1 | | |-- config2 | | `-- config3 | `-- fooapp -> castle/home/.config/fooapp `-- .emacs.d |-- elisp `-- inits Or `homesick track NESTED_FILE CASTLE` adds a line automatically. For example: homesick track .emacs.d/elisp castle castle/.homesick_subdir .config .emacs.d home directory ~ |-- .config | |-- barapp | | |-- config1 | | |-- config2 | | `-- config3 | `-- fooapp -> castle/home/.config/fooapp `-- .emacs.d |-- elisp -> castle/home/.emacs.d/elisp `-- inits and castle castle/home |-- .config | `-- fooapp | |-- config1 | |-- config2 | `-- config3 `-- .emacs.d `-- elisp ## Supported Ruby Versions Homesick is tested on the following Ruby versions: * 1.9.3 * 2.0.0 * 2.1.0 ## Note on Patches/Pull Requests * Fork the project. * Make your feature addition or bug fix. * Add tests for it. This is important so I don't break it in a future version unintentionally. * Commit, do not mess with rakefile, version, or history. (if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull) * Send me a pull request. Bonus points for topic branches. ## Need homesick without the ruby dependency? Check out [homeshick](https://github.com/andsens/homeshick). ## Copyright Copyright (c) 2010 Joshua Nichols. See LICENSE for details. homesick-1.1.3/.rubocop.yml0000644000004100000410000000067312642331233015625 0ustar www-datawww-data# TODO: Eval is required for the .homesickrc feature. This should eventually be # removed if the feature is implemented in a more secure way. Eval: Enabled: false # TODO: The following settings disable reports about issues that can be fixed # through refactoring. Remove these as offenses are removed from the code base. ClassLength: Enabled: false CyclomaticComplexity: Max: 13 LineLength: Enabled: false MethodLength: Max: 36 homesick-1.1.3/metadata.yml0000644000004100000410000001320312642331233015647 0ustar www-datawww-data--- !ruby/object:Gem::Specification name: homesick version: !ruby/object:Gem::Version version: 1.1.3 platform: ruby authors: - Joshua Nichols - Yusuke Murata autorequire: bindir: bin cert_chain: [] date: 2015-10-31 00:00:00.000000000 Z dependencies: - !ruby/object:Gem::Dependency name: thor requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 0.14.0 type: :runtime prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 0.14.0 - !ruby/object:Gem::Dependency name: rake requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 0.8.7 type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 0.8.7 - !ruby/object:Gem::Dependency name: rspec requirement: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 3.1.0 type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 3.1.0 - !ruby/object:Gem::Dependency name: guard requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' - !ruby/object:Gem::Dependency name: guard-rspec requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' - !ruby/object:Gem::Dependency name: rb-readline requirement: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 0.5.0 type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 0.5.0 - !ruby/object:Gem::Dependency name: jeweler requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 1.6.2 type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: 1.6.2 - !ruby/object:Gem::Dependency name: coveralls requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' - !ruby/object:Gem::Dependency name: test_construct requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' - !ruby/object:Gem::Dependency name: capture-output requirement: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 1.0.0 type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - "~>" - !ruby/object:Gem::Version version: 1.0.0 - !ruby/object:Gem::Dependency name: rubocop requirement: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' type: :development prerelease: false version_requirements: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' description: "\n Your home directory is your castle. Don't leave your dotfiles behind.\n \n\n Homesick is sorta like rip, but for dotfiles. It uses git to clone a repository containing dotfiles, and saves them in ~/.homesick. It then allows you to symlink all the dotfiles into place with a single command. \n\n " email: - josh@technicalpickles.com - info@muratayusuke.com executables: - homesick extensions: [] extra_rdoc_files: - ChangeLog.markdown - LICENSE - README.markdown files: - ".document" - ".rspec" - ".rubocop.yml" - ".travis.yml" - ChangeLog.markdown - Gemfile - Guardfile - LICENSE - README.markdown - Rakefile - bin/homesick - homesick.gemspec - lib/homesick.rb - lib/homesick/actions/file_actions.rb - lib/homesick/actions/git_actions.rb - lib/homesick/cli.rb - lib/homesick/utils.rb - lib/homesick/version.rb - spec/homesick_cli_spec.rb - spec/spec.opts - spec/spec_helper.rb homepage: http://github.com/technicalpickles/homesick licenses: - MIT metadata: {} post_install_message: rdoc_options: [] require_paths: - lib required_ruby_version: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' required_rubygems_version: !ruby/object:Gem::Requirement requirements: - - ">=" - !ruby/object:Gem::Version version: '0' requirements: [] rubyforge_project: rubygems_version: 2.4.8 signing_key: specification_version: 4 summary: Your home directory is your castle. Don't leave your dotfiles behind. test_files: [] has_rdoc: homesick-1.1.3/ChangeLog.markdown0000644000004100000410000000607112642331233016744 0ustar www-datawww-data#1.1.3 * Allow a destinaton to be passed when cloning a castle * Make sure `homesick edit` opens default editor in the root of the given castle * Fixed bug when diffing edited files * Fixed crashing bug when attempting to diff directories * Ensure that messages are escaped correctly on `git commit all` #1.1.2 * Added '--force' option to the rc command to bypass confirmation checks when running a .homesickrc file * Added a check to make sure that a minimum of Git 1.8.0 is installed. This stops Homesick failing silently if Git is not installed. * Code refactoring and fixes. #1.1.0 * Added exec and exec_all commands to run commands inside one or all clones castles. * Code refactoring. #1.0.0 * Removed support for Ruby 1.8.7 * Added a version command # 0.9.8 * Introduce new commands * `homesick cd` * `homesick open` # 0.9.4 * Use https protocol instead of git protocol * Introduce new commands * `homesick unlink` * `homesick rc` # 0.9.3 * Add recursive option to `homesick clone` # 0.9.2 * Set "dotfiles" as default castle name * Introduce new commands * `homesick show_path` * `homesick status` * `homesick diff` # 0.9.1 * Fixed small bugs: #35, #40 # 0.9.0 * Introduce .homesick_subdir #39 # 0.8.1 *Fixed `homesick list` bug on ruby 2.0 #37 # 0.8.0 * Introduce commit & push command * commit changes in castle and push to remote * Enable recursive submodule update * Git add when track # 0.7.0 * Fixed double-cloning #14 * New option for pull command: --all * pulls each castle, instead of just one # 0.6.1 * Add a license # 0.6.0 * Introduce .homesickrc * Castles can now have a .homesickrc inside them * On clone, this is eval'd inside the destination directory * Introduce track command * Allows easily moving an existing file into a castle, and symlinking it back # 0.5.0 * Fixed listing of castles cloned using `homesick clone /` (issue 3) * Added `homesick pull ` for updating castles (thanks Jorge Dias!) * Added a very basic `homesick generate ` # 0.4.1 * Improved error message when a castle's home dir doesn't exist # 0.4.0 * `homesick clone` can now take a path to a directory on the filesystem, which will be symlinked into place * `homesick clone` now tries to `git submodule init` and `git submodule update` if git submodules are defined for a cloned repo * Fixed missing dependency on thor and others * Use HOME environment variable for where to store files, instead of assuming ~ # 0.3.0 * Renamed 'link' to 'symlink' * Fixed conflict resolution when symlink destination exists and is a normal file # 0.2.0 * Better support for recognizing git urls (thanks jacobat!) * if it looks like a github user/repo, do that * otherwise hand off to git clone * Listing now displays in color, and show git remote * Support pretend, force, and quiet modes # 0.1.1 * Fixed trying to link against castles that don't exist * Fixed linking, which tries to exclude . and .. from the list of files to link (thanks Martinos!) # 0.1.0 * Initial release homesick-1.1.3/LICENSE0000644000004100000410000000204212642331233014350 0ustar www-datawww-dataCopyright (c) 2009 Joshua Nichols Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. homesick-1.1.3/.document0000644000004100000410000000007412642331233015165 0ustar www-datawww-dataREADME.rdoc lib/**/*.rb bin/* features/**/*.feature LICENSE homesick-1.1.3/Guardfile0000644000004100000410000000035012642331233015170 0ustar www-datawww-dataguard :rspec, :cmd => 'bundle exec rspec' do watch(%r{^spec/.+_spec\.rb$}) watch(%r{^lib/(.+)\.rb$}) { |m| "spec/#{m[1]}_spec.rb" } watch(%r{^lib/homesick/.*\.rb}) { "spec" } watch('spec/spec_helper.rb') { "spec" } end