Merge pull request #137 from johnduarte/git_ensure_latest
[puppet_vcsrepo.git] / spec / acceptance / clone_repo_spec.rb
1 require 'spec_helper_acceptance'
2
3 tmpdir = default.tmpdir('vcsrepo')
4
5 describe 'clones a remote repo' do
6   before(:all) do
7     my_root = File.expand_path(File.join(File.dirname(__FILE__), '..'))
8     shell("mkdir -p #{tmpdir}") # win test
9     scp_to(default, "#{my_root}/acceptance/files/create_git_repo.sh", tmpdir)
10     shell("cd #{tmpdir} && ./create_git_repo.sh")
11   end
12
13   after(:all) do
14     shell("rm -rf #{tmpdir}/testrepo.git")
15   end
16
17   context 'get the current master HEAD' do
18     it 'clones a repo' do
19       pp = <<-EOS
20       vcsrepo { "#{tmpdir}/testrepo":
21         ensure => present,
22         provider => git,
23         source => "file://#{tmpdir}/testrepo.git",
24       }
25       EOS
26
27       # Run it twice and test for idempotency
28       apply_manifest(pp, :catch_failures => true)
29       apply_manifest(pp, :catch_changes => true)
30     end
31
32     describe file("#{tmpdir}/testrepo/.git") do
33       it { should be_directory }
34     end
35
36     describe file("#{tmpdir}/testrepo/.git/HEAD") do
37       it { should contain 'ref: refs/heads/master' }
38     end
39   end
40
41   context 'using a commit SHA' do
42     let (:sha) do
43       shell("git --git-dir=#{tmpdir}/testrepo.git rev-list HEAD | tail -1").stdout.chomp
44     end
45
46     after(:all) do
47       shell("rm -rf #{tmpdir}/testrepo_sha")
48     end
49
50     it 'clones a repo' do
51       pp = <<-EOS
52       vcsrepo { "#{tmpdir}/testrepo_sha":
53         ensure => present,
54         provider => git,
55         source => "file://#{tmpdir}/testrepo.git",
56         revision => "#{sha}",
57       }
58       EOS
59
60       # Run it twice and test for idempotency
61       apply_manifest(pp, :catch_failures => true)
62       apply_manifest(pp, :catch_changes => true)
63     end
64
65     describe file("#{tmpdir}/testrepo_sha/.git") do
66       it { should be_directory }
67     end
68
69     describe file("#{tmpdir}/testrepo_sha/.git/HEAD") do
70       it { should contain sha }
71     end
72   end
73
74   context 'using a tag' do
75     it 'clones a repo' do
76       pp = <<-EOS
77       vcsrepo { "#{tmpdir}/testrepo_tag":
78         ensure => present,
79         provider => git,
80         source => "file://#{tmpdir}/testrepo.git",
81         revision => '0.0.2',
82       }
83       EOS
84
85       # Run it twice and test for idempotency
86       apply_manifest(pp, :catch_failures => true)
87       apply_manifest(pp, :catch_changes => true)
88     end
89
90     describe file("#{tmpdir}/testrepo_tag/.git") do
91       it { should be_directory }
92     end
93
94     it 'should have the tag as the HEAD' do
95       shell("git --git-dir=#{tmpdir}/testrepo_tag/.git name-rev HEAD | grep '0.0.2'")
96     end
97   end
98
99   context 'using a branch name' do
100     it 'clones a repo' do
101       pp = <<-EOS
102       vcsrepo { "#{tmpdir}/testrepo_branch":
103         ensure => present,
104         provider => git,
105         source => "file://#{tmpdir}/testrepo.git",
106         revision => 'a_branch',
107       }
108       EOS
109
110       # Run it twice and test for idempotency
111       apply_manifest(pp, :catch_failures => true)
112       apply_manifest(pp, :catch_changes => true)
113     end
114
115     describe file("#{tmpdir}/testrepo_branch/.git") do
116       it { should be_directory }
117     end
118
119     describe file("#{tmpdir}/testrepo_branch/.git/HEAD") do
120       it { should contain 'ref: refs/heads/a_branch' }
121     end
122   end
123
124   context 'ensure latest with branch specified' do
125     it 'clones a repo' do
126       pp = <<-EOS
127       vcsrepo { "#{tmpdir}/testrepo_latest":
128         ensure => latest,
129         provider => git,
130         source => "file://#{tmpdir}/testrepo.git",
131         revision => 'a_branch',
132       }
133       EOS
134
135       # Run it twice and test for idempotency
136       apply_manifest(pp, :catch_failures => true)
137       apply_manifest(pp, :catch_changes => true)
138     end
139
140     it 'verifies the HEAD commit SHA on remote and local match' do
141       remote_commit = shell("git ls-remote file://#{tmpdir}/testrepo_latest HEAD | head -1").stdout
142       local_commit = shell("git --git-dir=#{tmpdir}/testrepo_latest/.git rev-parse HEAD").stdout.chomp
143       expect(remote_commit).to include(local_commit)
144     end
145   end
146
147   context 'ensure latest with branch unspecified' do
148     it 'clones a repo' do
149       pp = <<-EOS
150       vcsrepo { "#{tmpdir}/testrepo_latest":
151         ensure => latest,
152         provider => git,
153         source => "file://#{tmpdir}/testrepo.git",
154       }
155       EOS
156
157       # Run it twice and test for idempotency
158       apply_manifest(pp, :catch_failures => true)
159       apply_manifest(pp, :catch_changes => true)
160     end
161
162     it 'verifies the HEAD commit SHA on remote and local match' do
163       remote_commit = shell("git ls-remote file://#{tmpdir}/testrepo_latest HEAD | head -1").stdout
164       local_commit = shell("git --git-dir=#{tmpdir}/testrepo_latest/.git rev-parse HEAD").stdout.chomp
165       expect(remote_commit).to include(local_commit)
166     end
167   end
168
169   context 'with shallow clone' do
170     it 'does a shallow clone' do
171       pp = <<-EOS
172       vcsrepo { "#{tmpdir}/testrepo_shallow":
173         ensure => present,
174         provider => git,
175         source => "file://#{tmpdir}/testrepo.git",
176         depth => '1',
177       }
178       EOS
179
180       # Run it twice and test for idempotency
181       apply_manifest(pp, :catch_failures => true)
182       apply_manifest(pp, :catch_changes => true)
183     end
184
185     describe file("#{tmpdir}/testrepo_shallow/.git/shallow") do
186       it { should be_file }
187     end
188   end
189
190   context 'path is not empty and not a repository' do
191     before(:all) do
192       shell("mkdir #{tmpdir}/not_a_repo", :acceptable_exit_codes => [0,1])
193       shell("touch #{tmpdir}/not_a_repo/file1.txt", :acceptable_exit_codes => [0,1])
194     end
195
196     it 'should raise an exception' do
197       pp = <<-EOS
198       vcsrepo { "#{tmpdir}/not_a_repo":
199         ensure => present,
200         provider => git
201         source => "file://#{tmpdir}/testrepo.git",
202       }
203       EOS
204       apply_manifest(pp, :expect_failures => true)
205     end
206   end
207
208   context 'with an owner' do
209     pp = <<-EOS
210     user { 'vagrant':
211       ensure => present,
212     }
213     EOS
214
215     apply_manifest(pp, :catch_failures => true)
216     it 'clones a repo' do
217       pp = <<-EOS
218       vcsrepo { "#{tmpdir}/testrepo_owner":
219         ensure => present,
220         provider => git,
221         source => "file://#{tmpdir}/testrepo.git",
222         owner => 'vagrant',
223       }
224       EOS
225
226       # Run it twice and test for idempotency
227       apply_manifest(pp, :catch_failures => true)
228       apply_manifest(pp, :catch_changes => true)
229     end
230
231     describe file("#{tmpdir}/testrepo_owner") do
232       it { should be_directory }
233       it { should be_owned_by 'vagrant' }
234     end
235   end
236
237   context 'with a group' do
238     pp = <<-EOS
239     group { 'vagrant':
240       ensure => present,
241     }
242     EOS
243
244     apply_manifest(pp, :catch_failures => true)
245
246     it 'clones a repo' do
247       pp = <<-EOS
248       vcsrepo { "/#{tmpdir}/testrepo_group":
249         ensure => present,
250         provider => git,
251         source => "file://#{tmpdir}/testrepo.git",
252         group => 'vagrant',
253       }
254       EOS
255
256       # Run it twice and test for idempotency
257       apply_manifest(pp, :catch_failures => true)
258       apply_manifest(pp, :catch_changes => true)
259     end
260
261     describe file("#{tmpdir}/testrepo_group") do
262       it { should be_directory }
263       it { should be_grouped_into 'vagrant' }
264     end
265   end
266
267   context 'with excludes' do
268     it 'clones a repo' do
269       pp = <<-EOS
270       vcsrepo { "#{tmpdir}/testrepo_excludes":
271         ensure => present,
272         provider => git,
273         source => "file://#{tmpdir}/testrepo.git",
274         excludes => ['exclude1.txt', 'exclude2.txt'],
275       }
276       EOS
277
278       # Run it twice and test for idempotency
279       apply_manifest(pp, :catch_failures => true)
280       apply_manifest(pp, :catch_changes => true)
281     end
282
283     describe file("#{tmpdir}/testrepo_excludes/.git/info/exclude") do
284       its(:content) { should match /exclude1.txt/ }
285       its(:content) { should match /exclude2.txt/ }
286     end
287   end
288
289   context 'with force' do
290     before(:all) do
291       shell("mkdir -p #{tmpdir}/testrepo_force/folder")
292       shell("touch #{tmpdir}/testrepo_force/temp.txt")
293     end
294
295     it 'applies the manifest' do
296       pp = <<-EOS
297       vcsrepo { "#{tmpdir}/testrepo_force":
298         ensure => present,
299         provider => git,
300         source => "file://#{tmpdir}/testrepo.git",
301         force => true,
302       }
303       EOS
304
305       # Run it twice and test for idempotency
306       apply_manifest(pp, :catch_failures => true)
307       apply_manifest(pp, :catch_changes => true)
308     end
309
310     describe file("#{tmpdir}/testrepo_force/folder") do
311       it { should_not be_directory }
312     end
313
314     describe file("#{tmpdir}/testrepo_force/temp.txt") do
315       it { should_not be_file }
316     end
317
318     describe file("#{tmpdir}/testrepo_force/.git") do
319       it { should be_directory }
320     end
321   end
322
323   context 'as a user' do
324     before(:all) do
325       shell("chmod 707 #{tmpdir}")
326       pp = <<-EOS
327       user { 'testuser':
328         ensure => present,
329       }
330       EOS
331
332       apply_manifest(pp, :catch_failures => true)
333     end
334
335     it 'applies the manifest' do
336       pp = <<-EOS
337       vcsrepo { "#{tmpdir}/testrepo_user":
338         ensure => present,
339         provider => git,
340         source => "file://#{tmpdir}/testrepo.git",
341         user => 'testuser',
342       }
343       EOS
344
345       # Run it twice and test for idempotency
346       apply_manifest(pp, :catch_failures => true)
347       apply_manifest(pp, :catch_changes => true)
348     end
349
350     describe file("#{tmpdir}/testrepo_user") do
351       it { should be_directory }
352       it { should be_owned_by 'testuser' }
353     end
354
355     describe file("#{tmpdir}/testrepo_user") do
356       it { should be_directory }
357       it { should be_grouped_into 'testuser' }
358     end
359   end
360
361   context 'non-origin remote name' do
362     it 'applies the manifest' do
363       pp = <<-EOS
364       vcsrepo { "#{tmpdir}/testrepo_remote":
365         ensure => present,
366         provider => git,
367         source => "file://#{tmpdir}/testrepo.git",
368         remote => 'testorigin',
369       }
370       EOS
371
372       # Run it twice and test for idempotency
373       apply_manifest(pp, :catch_failures => true)
374       apply_manifest(pp, :catch_changes => true)
375     end
376
377     it 'remote name is "testorigin"' do
378       shell("git --git-dir=#{tmpdir}/testrepo_remote/.git remote | grep 'testorigin'")
379     end
380
381     after(:all) do
382       pp = 'user { "testuser": ensure => absent }'
383       apply_manifest(pp, :catch_failures => true)
384     end
385   end
386
387   context 'as a user with ssh' do
388     before(:all) do
389       # create user
390       pp = <<-EOS
391       user { 'testuser-ssh':
392         ensure => present,
393         managehome => true,
394       }
395       EOS
396       apply_manifest(pp, :catch_failures => true)
397
398       # create ssh keys
399       shell('mkdir -p /home/testuser-ssh/.ssh')
400       shell('ssh-keygen -q -t rsa -f /home/testuser-ssh/.ssh/id_rsa -N ""')
401
402       # copy public key to authorized_keys
403       shell('cat /home/testuser-ssh/.ssh/id_rsa.pub > /home/testuser-ssh/.ssh/authorized_keys')
404       shell('echo -e "Host localhost\n\tStrictHostKeyChecking no\n" > /home/testuser-ssh/.ssh/config')
405       shell('chown -R testuser-ssh:testuser-ssh /home/testuser-ssh/.ssh')
406     end
407
408     it 'applies the manifest' do
409       pp = <<-EOS
410       vcsrepo { "#{tmpdir}/testrepo_user_ssh":
411         ensure => present,
412         provider => git,
413         source => "testuser-ssh@localhost:#{tmpdir}/testrepo.git",
414         user => 'testuser-ssh',
415       }
416       EOS
417
418       # Run it twice and test for idempotency
419       apply_manifest(pp, :catch_failures => true)
420       apply_manifest(pp, :catch_changes => true)
421     end
422
423     after(:all) do
424       pp = <<-EOS
425       user { 'testuser-ssh':
426         ensure => absent,
427         managehome => true,
428       }
429       EOS
430       apply_manifest(pp, :catch_failures => true)
431     end
432   end
433
434   context 'using an identity file' do
435     before(:all) do
436       # create user
437       pp = <<-EOS
438       user { 'testuser-ssh':
439         ensure => present,
440         managehome => true,
441       }
442       EOS
443       apply_manifest(pp, :catch_failures => true)
444
445       # create ssh keys
446       shell('mkdir -p /home/testuser-ssh/.ssh')
447       shell('ssh-keygen -q -t rsa -f /home/testuser-ssh/.ssh/id_rsa -N ""')
448
449       # copy public key to authorized_keys
450       shell('cat /home/testuser-ssh/.ssh/id_rsa.pub > /home/testuser-ssh/.ssh/authorized_keys')
451       shell('echo -e "Host localhost\n\tStrictHostKeyChecking no\n" > /home/testuser-ssh/.ssh/config')
452       shell('chown -R testuser-ssh:testuser-ssh /home/testuser-ssh/.ssh')
453     end
454
455     it 'applies the manifest' do
456       pp = <<-EOS
457       vcsrepo { "#{tmpdir}/testrepo_user_ssh_id":
458         ensure => present,
459         provider => git,
460         source => "testuser-ssh@localhost:#{tmpdir}/testrepo.git",
461         identity => '/home/testuser-ssh/.ssh/id_rsa',
462       }
463       EOS
464
465       # Run it twice and test for idempotency
466       apply_manifest(pp, :catch_failures => true)
467       apply_manifest(pp, :catch_changes => true)
468     end
469
470     after(:all) do
471       pp = <<-EOS
472       user { 'testuser-ssh':
473         ensure => absent,
474         managehome => true,
475       }
476       EOS
477       apply_manifest(pp, :catch_failures => true)
478     end
479   end
480 end