pact-retreaty 0.2.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,15 @@
1
+ ---
2
+ !binary "U0hBMQ==":
3
+ metadata.gz: !binary |-
4
+ Yjk5MTk0NWQ5ZTY2YmUwZWRhYTJmYjVkNTU2NDg5ZjAwOWQ0MmU0ZA==
5
+ data.tar.gz: !binary |-
6
+ ODVlOWEzNjQyOGNmYWY0NDMwOTk0OTE0NzgxZjE5NjZhODJiMDZmMw==
7
+ SHA512:
8
+ metadata.gz: !binary |-
9
+ ZmFhYjBlMmRiNTBlMGI4NTFhZWEyYjc4MWUzODNlZDUzMThjYTUzZWZiMjQ5
10
+ NjllMjAxODk0ZjAxNmNhM2U4ZTkxZTIzNjA1MDEyYzQyMzExMzkwNjhjYWM2
11
+ NTM0ZTFjYTBjOTIyNDlhMjU0MGJhMTAwODQwYzc2OTU0MTk1Y2I=
12
+ data.tar.gz: !binary |-
13
+ ZGYzZGEyMDVhMjY2MWNkNjAwNWZiYzVlMGRhMmRhNTRmYzdmYzYxYjEzNjA1
14
+ ZWZiNTVhNWEzNGM4ZjUxOWI2YTI5NTUxMDQyNzFmNjExMWQ1YjQ0YTcwZTE2
15
+ YWUyNTc2OTA2YzJhN2U1OGI5MjJmM2U3YTEzNjYxY2ZlYjcwMjU=
@@ -0,0 +1,11 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /Gemfile.lock
4
+ /_yardoc/
5
+ /coverage/
6
+ /doc/
7
+ /pkg/
8
+ /spec/reports/
9
+ /tmp/
10
+ defaults.yml
11
+ *.gem
data/.rspec ADDED
@@ -0,0 +1,2 @@
1
+ --format documentation
2
+ --color
@@ -0,0 +1,4 @@
1
+ language: ruby
2
+ rvm:
3
+ - 1.9.3
4
+ before_install: gem install bundler -v 1.10.5
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in pact-retreaty.gemspec
4
+ gemspec
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2015 Fairfax Media
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
@@ -0,0 +1,108 @@
1
+ # Pact::Retreaty
2
+
3
+ [Pact](http://github.com/pact/pact) allows the recording of contracts between services and their consumers, but is agnostic about how those contracts are managed. [Retreaty](http://github.com/fairfacemedia/pact-retreaty) extends on that, to provides a ultra light mechanism for pushing these contracts to S3 from a consumer, and later pulling them down to a provider for verification.
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ ```ruby
10
+ gem 'pact-retreaty'
11
+ ```
12
+
13
+ And then execute:
14
+
15
+ $ bundle
16
+
17
+ Or install it yourself as:
18
+
19
+ $ gem install pact-retreaty
20
+
21
+ ## Usage
22
+
23
+ ### Consumer-side
24
+
25
+ Once you've got Pact installed and working, copy this rake task into your Rakefile:
26
+
27
+ ```ruby
28
+ namespace :pact do
29
+ desc "Upload pact contracts to S3"
30
+ task :upload do
31
+ require_relative 'spec/service_providers/pact_helper'
32
+ require 'pact/retreaty'
33
+
34
+ Pact::Retreaty::Consumer.create do |consumer|
35
+ consumer.name = 'ds-spockly'
36
+ consumer.version = Spockly::VERSION
37
+ consumer.source_glob = "#{Pact.configuration.pact_dir}/*.json"
38
+ consumer.vcs_id = -> { ENV['CI'] ? ENV['CI_BRANCH'] : current_vcs_id }
39
+ consumer.s3_bucket = ENV['PACT_S3_BUCKET']
40
+ consumer.s3_region = ENV['PACT_S3_REGION']
41
+ consumer.access_key_id = ENV['PACT_S3_KEY_ID']
42
+ consumer.access_secret = ENV['PACT_S3_SECRET']
43
+ end.upload_pacts
44
+ end
45
+ end
46
+ ```
47
+
48
+ Replace the configuration with what makes sense in your consumer - the name and version associated with the current consumer, and an optional lambda to identify the applicable branch from you VCS (Retreaty defaults to Git, and to the branch of the repo it's being run in.)
49
+
50
+ (It's assumed that this task will usually be running on your CI server, and that you'll set the S3 credentials via environment variables - but you can tweak that behaviour here if you need to.)
51
+
52
+ When run, the task will load your Pact configuration from your pact_helper file and try to upload all the json files it finds in spec/pacts to the bucket defined. The path inside the bucket will look something like (based on the configuration above, assuming that Spockly::VERSION is '1.0.0' the current branch is 'pact-testing' and the pact is between "Spockly Gem" and "Spock Service"):
53
+
54
+ ```
55
+ /ds-spockly/1.0.0/pact-testing/spockly_gem-spock_service.json
56
+ ```
57
+
58
+ ### Provider-side
59
+
60
+ The configuration on the provider side looks very similar to the consumer side. Once you've got Pact running with a local contract file, change your pact_helper file to look something like this:
61
+
62
+ ```ruby
63
+ require 'pact/retreaty'
64
+
65
+ Pact.service_provider "Spock Service" do
66
+ honours_pact_with 'Spockly Gem' do
67
+ if ENV['CI'] # CI server
68
+ pact_uri = Pact::Retreaty::Consumer.create do |consumer|
69
+ consumer.name = 'ds-spockly'
70
+ consumer.version = '1.0.0'
71
+ consumer.pactfile = 'spockly_gem-spock_service.json'
72
+ consumer.vcs_fallbacks = -> { [ENV['CI_BRANCH'], :vcs_id, 'develop'] }
73
+ consumer.s3_bucket = ENV['PACT_S3_BUCKET']
74
+ consumer.s3_region = ENV['PACT_S3_REGION']
75
+ consumer.access_key_id = ENV['PACT_S3_KEY_ID']
76
+ consumer.access_secret = ENV['PACT_S3_SECRET']
77
+ end.best_pact_uri
78
+ else # local
79
+ pact_uri '../ds-spockly/spec/pacts/spockly_gem-spock_service.json'
80
+ end
81
+ end
82
+ end
83
+ ```
84
+
85
+ (In the example above, we're setting the pact_uri based on whether we're running on a CI server - if we're not, we fall back to a local directory as usual.)
86
+
87
+ The name, version and S3 configurations are identical to the consumer side - the only provider-specific configuration is that we help Retreaty find the right contract by specifying the filename, and we provide a series of options to find the right branch. This deserves some explanation...
88
+
89
+ ###VCS Fallbacks
90
+
91
+ Over the lifetime of our Pact testing, we'll want to verify various versions of the consumer against a variety of versions of the provider. We have a few mechanisms in place to facilitate this; on the consumer side we track both the 'version' and the 'VCS id' (version control id - usually a branch name) of the current build, and on the provider side we keep a list of options (in order of decreasing specificity) for which of the consumer's VCS ids we'd like to test against. Tracking the branch of the consumer protects us while we're modifying it, so that new versions of the contract don't overwrite the existing stable one - it also frees us from having to change the version for small bug fixes, or during the code review process.
92
+
93
+ On the consumer side, we can branch in lockstep with the consumer, and then have pact verification before we submit our changes for review - but we can pick a default vcs id ('develop', in the above example) and always fall back to that version of the consumer as a verification target. (Retreaty will try each option in the fallbacks list until it's able to identify and stored contract in S3.)
94
+
95
+ ## Development
96
+
97
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake rspec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
98
+
99
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
100
+
101
+ ## Contributing
102
+
103
+ Bug reports and pull requests are welcome on GitHub at https://github.com/fairfaxmedia/pact-retreaty.
104
+
105
+
106
+ ## License
107
+
108
+ The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
@@ -0,0 +1,26 @@
1
+ require "bundler/gem_tasks"
2
+ require "rspec/core/rake_task"
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+
6
+ task :default => :spec
7
+
8
+
9
+ task :console do
10
+ require 'irb'
11
+ require 'irb/completion'
12
+ require 'pact/retreaty'
13
+ require 'pry'
14
+
15
+ defaults_path = File.expand_path('../defaults.yml', __FILE__)
16
+ defaults = File.exists?(defaults_path) ? YAML.load_file(defaults_path) : {}
17
+
18
+ Pact::Retreaty.define_singleton_method(:default_consumer) {
19
+ Pact::Retreaty::Consumer.create do |consumer|
20
+ defaults.each { |k, v| consumer.send("#{k}=", v) }
21
+ end #.best_pact_uri
22
+ }
23
+
24
+ ARGV.clear
25
+ IRB.start
26
+ end
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "bundler/setup"
4
+ require "pact/retreaty"
5
+
6
+ # You can add fixtures and/or initialization code here to make experimenting
7
+ # with your gem easier. You can also use a different console, if you like.
8
+
9
+ # (If you use this, don't forget to add pry to your Gemfile!)
10
+ # require "pry"
11
+ # Pry.start
12
+
13
+ require "irb"
14
+ IRB.start
@@ -0,0 +1,7 @@
1
+ #!/bin/bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+
5
+ bundle install
6
+
7
+ # Do any other automated setup that you need to do here
@@ -0,0 +1,7 @@
1
+ require "pact/retreaty/version"
2
+ require 'pact/retreaty/consumer'
3
+
4
+ module Pact
5
+ module Retreaty
6
+ end
7
+ end
@@ -0,0 +1,98 @@
1
+ require 'aws-sdk'
2
+ require 'pact'
3
+
4
+ module Pact
5
+ module Retreaty
6
+ class Consumer
7
+ COMMON_CONFIG = [:name, :version, :s3_bucket, :s3_region, :access_key_id, :access_secret]
8
+ UPLOAD_CONFIG = [:vcs_id]
9
+ DOWNLOAD_CONFIG = [:pactfile, :vcs_fallbacks]
10
+ attr_accessor *(COMMON_CONFIG + UPLOAD_CONFIG + DOWNLOAD_CONFIG)
11
+
12
+ def self.create(options = {})
13
+ options[:vcs_id] ||= -> { current_vcs_id }
14
+ new(options).tap { |consumer| yield consumer if block_given? }
15
+ end
16
+
17
+ def upload_pacts
18
+ verify_config!(:upload)
19
+
20
+ spec_files.each do |path|
21
+ key = upload_key_for_path(path)
22
+ puts "uploading #{key}"
23
+ s3_client.put_object(bucket: s3_bucket, key: key, body: File.read(path))
24
+ end
25
+ end
26
+
27
+ def best_pact_uri
28
+ verify_config!(:download)
29
+
30
+ catch(:uri_found) do
31
+ realised_vcs_fallbacks.each { |vcs_id| uri_for_pactfile_and_vcs_id(vcs_id) }
32
+ fail("Retreaty couldn't find a suitable contract for version #{version} of #{name}, under #{realised_vcs_fallbacks.join(' or ')}")
33
+ end
34
+ end
35
+
36
+ private
37
+
38
+ def initialize(options = {})
39
+ options.each do |k, v|
40
+ send("#{k}=", v)
41
+ end
42
+ end
43
+
44
+ def uri_for_pactfile_and_vcs_id(vcs_id)
45
+ summary = s3_summary_for_vcs_id(vcs_id)
46
+ throw(:uri_found, summary.presigned_url(:get)) if summary.exists?
47
+ end
48
+
49
+ def s3_summary_for_vcs_id(vcs_id)
50
+ Aws::S3::ObjectSummary.new(bucket_name: s3_bucket, key: s3_path(pactfile, vcs_id), client: s3_client)
51
+ end
52
+
53
+ def verify_config!(context)
54
+ required_fields = COMMON_CONFIG + (context == :download ? DOWNLOAD_CONFIG : UPLOAD_CONFIG)
55
+ unless required_fields.select {|field| send(field).nil? }.empty?
56
+ fail("Retreaty requires configuration for #{required_fields.join(', ')} to #{context}")
57
+ end
58
+ end
59
+
60
+ def spec_files
61
+ Dir.glob(File.join(pact_dir, '*.json'))
62
+ end
63
+
64
+ def upload_key_for_path(path)
65
+ s3_path(File.basename(path), upload_vcs_id)
66
+ end
67
+
68
+ def upload_vcs_id
69
+ instance_exec(&vcs_id)
70
+ end
71
+
72
+ def s3_path(filename, vcs_id)
73
+ [name, version, vcs_id, filename].compact.join('/')
74
+ end
75
+
76
+ def realised_vcs_fallbacks
77
+ vcs_fallbacks.map {|id| id == :vcs_id ? current_vcs_id : id }
78
+ end
79
+
80
+ # TODO - handle cases where git isn't present - configure with a proc?
81
+ def current_vcs_id #current_branch
82
+ `git symbolic-ref -q --short HEAD`.strip
83
+ end
84
+
85
+ def pact_dir
86
+ Pact.configuration.pact_dir
87
+ end
88
+
89
+ def s3_client
90
+ @client ||= Aws::S3::Client.new(region: s3_region, credentials: s3_credentials)
91
+ end
92
+
93
+ def s3_credentials
94
+ Aws::Credentials.new(access_key_id, access_secret)
95
+ end
96
+ end
97
+ end
98
+ end
@@ -0,0 +1,5 @@
1
+ module Pact
2
+ module Retreaty
3
+ VERSION = "0.2.0"
4
+ end
5
+ end
@@ -0,0 +1,27 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'pact/retreaty/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "pact-retreaty"
8
+ spec.version = Pact::Retreaty::VERSION
9
+ spec.authors = ["Simon Hildebrandt"]
10
+ spec.email = ["simonhildebrandt@fairfaxmedia.com.au"]
11
+
12
+ spec.summary = %q{Easily share pacts via S3.}
13
+ spec.description = %q{Easily share pacts via S3.}
14
+ spec.homepage = "https://github.com/fairfaxmedia/pact-retreaty"
15
+ spec.license = "MIT"
16
+ spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
17
+ spec.bindir = "exe"
18
+ spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
19
+ spec.require_paths = ["lib"]
20
+
21
+ spec.add_dependency('aws-sdk', '~> 2')
22
+
23
+ spec.add_development_dependency "bundler", "~> 1.10"
24
+ spec.add_development_dependency "rake", "~> 10.0"
25
+ spec.add_development_dependency "rspec"
26
+ spec.add_development_dependency "pry"
27
+ end
metadata ADDED
@@ -0,0 +1,128 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: pact-retreaty
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.2.0
5
+ platform: ruby
6
+ authors:
7
+ - Simon Hildebrandt
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2016-04-06 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: aws-sdk
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ~>
18
+ - !ruby/object:Gem::Version
19
+ version: '2'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ~>
25
+ - !ruby/object:Gem::Version
26
+ version: '2'
27
+ - !ruby/object:Gem::Dependency
28
+ name: bundler
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ~>
32
+ - !ruby/object:Gem::Version
33
+ version: '1.10'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ~>
39
+ - !ruby/object:Gem::Version
40
+ version: '1.10'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rake
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ~>
46
+ - !ruby/object:Gem::Version
47
+ version: '10.0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ~>
53
+ - !ruby/object:Gem::Version
54
+ version: '10.0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rspec
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ! '>='
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ! '>='
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: pry
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ! '>='
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ! '>='
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ description: Easily share pacts via S3.
84
+ email:
85
+ - simonhildebrandt@fairfaxmedia.com.au
86
+ executables: []
87
+ extensions: []
88
+ extra_rdoc_files: []
89
+ files:
90
+ - .gitignore
91
+ - .rspec
92
+ - .travis.yml
93
+ - Gemfile
94
+ - LICENSE.txt
95
+ - README.md
96
+ - Rakefile
97
+ - bin/console
98
+ - bin/setup
99
+ - lib/pact/retreaty.rb
100
+ - lib/pact/retreaty/consumer.rb
101
+ - lib/pact/retreaty/version.rb
102
+ - pact-retreaty.gemspec
103
+ homepage: https://github.com/fairfaxmedia/pact-retreaty
104
+ licenses:
105
+ - MIT
106
+ metadata: {}
107
+ post_install_message:
108
+ rdoc_options: []
109
+ require_paths:
110
+ - lib
111
+ required_ruby_version: !ruby/object:Gem::Requirement
112
+ requirements:
113
+ - - ! '>='
114
+ - !ruby/object:Gem::Version
115
+ version: '0'
116
+ required_rubygems_version: !ruby/object:Gem::Requirement
117
+ requirements:
118
+ - - ! '>='
119
+ - !ruby/object:Gem::Version
120
+ version: '0'
121
+ requirements: []
122
+ rubyforge_project:
123
+ rubygems_version: 2.4.7
124
+ signing_key:
125
+ specification_version: 4
126
+ summary: Easily share pacts via S3.
127
+ test_files: []
128
+ has_rdoc: