-
-
Notifications
You must be signed in to change notification settings - Fork 235
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: Gamaliel Amaudruz <[email protected]>
- Loading branch information
Alan Yeo
authored and
Pair
committed
Dec 27, 2016
1 parent
5e67861
commit 137094e
Showing
16 changed files
with
348 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
require 'bundler' | ||
require 'yaml' | ||
require_relative '../../../lib/config/integrations/helpers/cf_manifest_merger' | ||
|
||
module Config | ||
module Integrations | ||
class CloudFoundry < Struct.new(:app_name, :file_path) | ||
|
||
def invoke | ||
manifest_path = file_path || 'manifest.yml' | ||
file_name, _ext = manifest_path.split('.yml') | ||
|
||
manifest_hash = YAML.load(IO.read(File.join(::Rails.root, manifest_path))) | ||
|
||
puts "Generating manifest... (base cf manifest: #{manifest_path})" | ||
|
||
merged_hash = Config::CFManifestMerger.new(app_name, manifest_hash).add_to_env | ||
|
||
target_manifest_path = File.join(::Rails.root, "#{file_name}-#{::Rails.env}.yml") | ||
IO.write(target_manifest_path, merged_hash.to_yaml) | ||
|
||
puts "File #{target_manifest_path} generated." | ||
end | ||
|
||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
require_relative 'helpers' | ||
|
||
module Config | ||
class CFManifestMerger | ||
include Integrations::Helpers | ||
|
||
def initialize(app_name, manifest_hash) | ||
@app_name = app_name | ||
@manifest_hash = manifest_hash | ||
raise ArgumentError.new("Manifest path & app name must be specified") unless @app_name && @manifest_hash | ||
end | ||
|
||
def add_to_env | ||
|
||
settings_hash = Config.const_get(Config.const_name).to_hash.stringify_keys | ||
|
||
prefix_keys_with_const_name_hash = to_dotted_hash(settings_hash, namespace: Config.const_name) | ||
|
||
app_hash = @manifest_hash['applications'].detect { |hash| hash['name'] == @app_name } | ||
|
||
raise ArgumentError, "Application '#{@app_name}' is not specified in your manifest" if app_hash.nil? | ||
|
||
check_conflicting_keys(app_hash['env'], settings_hash) | ||
|
||
app_hash['env'].merge!(prefix_keys_with_const_name_hash) | ||
|
||
@manifest_hash | ||
end | ||
|
||
private | ||
|
||
def check_conflicting_keys(env_hash, settings_hash) | ||
conflicting_keys = env_hash.keys & settings_hash.keys | ||
raise ArgumentError.new("Conflicting keys: #{conflicting_keys.join(', ')}") if conflicting_keys.any? | ||
end | ||
|
||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
module Config::Integrations::Helpers | ||
|
||
def to_dotted_hash(source, target: {}, namespace: nil) | ||
raise ArgumentError, "target must be a hash (given: #{target.class.name})" unless target.kind_of? Hash | ||
prefix = "#{namespace}." if namespace | ||
case source | ||
when Hash | ||
source.each do |key, value| | ||
to_dotted_hash(value, target: target, namespace: "#{prefix}#{key}") | ||
end | ||
when Array | ||
source.each_with_index do |value, index| | ||
to_dotted_hash(value, target: target, namespace: "#{prefix}#{index}") | ||
end | ||
else | ||
target[namespace] = source | ||
end | ||
target | ||
end | ||
|
||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
require 'config/integrations/cloud_foundry' | ||
|
||
namespace 'config' do | ||
|
||
desc 'Create a cf manifest with the env variables defined by config under current environment' | ||
task :'cf', [:app_name, :file_path] => :environment do |_, args| | ||
Config::Integrations::CloudFoundry.new(args[:app_name], args[:file_path]).invoke | ||
end | ||
|
||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,10 @@ | ||
require 'config/integrations/heroku' | ||
|
||
namespace 'config' do | ||
|
||
desc 'Upload to Heroku all env variables defined by config under current environment' | ||
task :heroku, [:app] => :environment do |_, args| | ||
Config::Integrations::Heroku.new(args[:app]).invoke | ||
end | ||
|
||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
DEFAULT_HOST: host | ||
DEFAULT_PORT: port |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
applications: | ||
- name: some-cf-app | ||
instances: 1 | ||
env: | ||
DEFAULT_HOST: host | ||
DEFAULT_PORT: port | ||
FOO: BAR | ||
|
||
- name: app_name | ||
env: | ||
DEFAULT_HOST: host |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
world: | ||
capitals: | ||
europe: | ||
germany: 'Berlin' | ||
poland: 'Warsaw' | ||
array: | ||
- name: 'Alan' | ||
- name: 'Gam' | ||
array_with_index: | ||
0: | ||
name: 'Bob' | ||
1: | ||
name: 'William' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,69 @@ | ||
require 'spec_helper' | ||
require_relative '../../../lib/config/integrations/helpers/cf_manifest_merger' | ||
|
||
describe Config::CFManifestMerger do | ||
|
||
after do | ||
Settings.reload_from_files("#{fixture_path}/settings.yml") | ||
end | ||
|
||
it 'raises an argument error if you do not specify an app name' do | ||
expect { | ||
Config::CFManifestMerger.new(nil, load_manifest('cf_manifest.yml')) | ||
}.to raise_error(ArgumentError, 'Manifest path & app name must be specified') | ||
end | ||
|
||
it 'raises an argument error if the application name is not found in the manifest' do | ||
expect { | ||
Config::CFManifestMerger.new('undefined', load_manifest('cf_manifest.yml')).add_to_env | ||
}.to raise_error(ArgumentError, "Application 'undefined' is not specified in your manifest") | ||
end | ||
|
||
it 'returns the cf manifest template if no settings available' do | ||
merger = Config::CFManifestMerger.new('app_name', load_manifest('cf_manifest.yml')) | ||
Config.load_and_set_settings '' | ||
|
||
resulting_hash = merger.add_to_env | ||
expect(resulting_hash).to eq(load_manifest('cf_manifest.yml')) | ||
end | ||
|
||
it 'merges the given YAML file with the cf manifest YAML file' do | ||
merger = Config::CFManifestMerger.new('some-cf-app', load_manifest('cf_manifest.yml')) | ||
Config.load_and_set_settings "#{fixture_path}/cf/cf_multilevel.yml" | ||
|
||
resulting_hash = merger.add_to_env | ||
expect(resulting_hash).to eq({ | ||
"applications" => [ | ||
{ | ||
"name" => "some-cf-app", | ||
"instances" => 1, | ||
"env" => { | ||
"DEFAULT_HOST" => "host", | ||
"DEFAULT_PORT" => "port", | ||
"FOO" => "BAR", | ||
"Settings.world.capitals.europe.germany" => "Berlin", | ||
"Settings.world.capitals.europe.poland" => "Warsaw", | ||
"Settings.world.array.0.name" => "Alan", | ||
"Settings.world.array.1.name" => "Gam", | ||
"Settings.world.array_with_index.0.name" => "Bob", | ||
"Settings.world.array_with_index.1.name" => "William" | ||
} | ||
}, | ||
{"name"=>"app_name", "env"=>{"DEFAULT_HOST"=>"host"}} | ||
] | ||
}) | ||
end | ||
|
||
it 'raises an exception if there is conflicting keys' do | ||
merger = Config::CFManifestMerger.new('some-cf-app', load_manifest('cf_manifest.yml')) | ||
Config.load_and_set_settings "#{fixture_path}/cf/cf_conflict.yml" | ||
|
||
expect { | ||
merger.add_to_env | ||
}.to raise_error(ArgumentError, 'Conflicting keys: DEFAULT_HOST, DEFAULT_PORT') | ||
end | ||
|
||
def load_manifest filename | ||
YAML.load(IO.read("#{fixture_path}/cf/#{filename}")) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
require 'spec_helper' | ||
require_relative '../../../lib/config/integrations/helpers/helpers' | ||
|
||
describe 'Helpers' do | ||
|
||
subject { Class.new.send(:include, Config::Integrations::Helpers).new } | ||
|
||
describe '#to_dotted_hash' do | ||
|
||
context 'only the source is specified' do | ||
|
||
it 'returns a hash with a nil key (default)' do | ||
expect(subject.to_dotted_hash 3).to eq({nil => 3}) | ||
end | ||
end | ||
|
||
context 'with invalid arguments' do | ||
it 'raises an error' do | ||
expect { subject.to_dotted_hash(3, target: [1, 2, 7], namespace: 2) } | ||
.to raise_error(ArgumentError, 'target must be a hash (given: Array)') | ||
end | ||
end | ||
|
||
context 'all arguments specified' do | ||
|
||
it 'returns a hash with the namespace as the key' do | ||
expect(subject.to_dotted_hash(3, namespace: 'ns')).to eq({'ns' => 3}) | ||
end | ||
|
||
it 'returns a new hash with a dotted string key prefixed with namespace' do | ||
expect(subject.to_dotted_hash({hello: {cruel: 'world'}}, namespace: 'ns')) | ||
.to eq({'ns.hello.cruel' => 'world'}) | ||
end | ||
|
||
it 'returns the same hash as passed as a parameter' do | ||
target = {something: 'inside'} | ||
target_id = target.object_id | ||
result = subject.to_dotted_hash(2, target: target, namespace: 'ns') | ||
expect(result).to eq({:something => 'inside', 'ns' => 2}) | ||
expect(result.object_id).to eq target_id | ||
end | ||
|
||
it 'returns a hash when given a source with mixed nested types (hashes & arrays)' do | ||
expect(subject.to_dotted_hash( | ||
{hello: {evil: [:cruel, 'world', and: {dark: 'universe'}]}}, namespace: 'ns')) | ||
.to eq( | ||
{"ns.hello.evil.0" => :cruel, | ||
"ns.hello.evil.1" => "world", | ||
"ns.hello.evil.2.and.dark" => "universe"} | ||
) | ||
end | ||
end | ||
end | ||
end |
Oops, something went wrong.