fog-serverlove 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (56) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +23 -0
  3. data/.hound.yml +20 -0
  4. data/.rubocop.yml +20 -0
  5. data/.ruby-gemset +1 -0
  6. data/.ruby-version +1 -0
  7. data/.travis.yml +17 -0
  8. data/CONTRIBUTING.md +18 -0
  9. data/CONTRIBUTORS.md +7 -0
  10. data/Gemfile +4 -0
  11. data/LICENSE.md +20 -0
  12. data/README.md +38 -0
  13. data/Rakefile +18 -0
  14. data/fog-serverlove.gemspec +35 -0
  15. data/gemfiles/Gemfile.1.9.2- +8 -0
  16. data/gemfiles/Gemfile.1.9.3+ +7 -0
  17. data/lib/fog/compute/serverlove.rb +95 -0
  18. data/lib/fog/compute/serverlove/models/image.rb +52 -0
  19. data/lib/fog/compute/serverlove/models/images.rb +19 -0
  20. data/lib/fog/compute/serverlove/models/server.rb +67 -0
  21. data/lib/fog/compute/serverlove/models/servers.rb +19 -0
  22. data/lib/fog/compute/serverlove/password_generator.rb +11 -0
  23. data/lib/fog/compute/serverlove/requests/create_image.rb +28 -0
  24. data/lib/fog/compute/serverlove/requests/create_server.rb +30 -0
  25. data/lib/fog/compute/serverlove/requests/destroy_image.rb +11 -0
  26. data/lib/fog/compute/serverlove/requests/destroy_server.rb +11 -0
  27. data/lib/fog/compute/serverlove/requests/get_image.rb +11 -0
  28. data/lib/fog/compute/serverlove/requests/get_images.rb +11 -0
  29. data/lib/fog/compute/serverlove/requests/get_server.rb +11 -0
  30. data/lib/fog/compute/serverlove/requests/get_servers.rb +11 -0
  31. data/lib/fog/compute/serverlove/requests/load_standard_image.rb +11 -0
  32. data/lib/fog/compute/serverlove/requests/reset_server.rb +11 -0
  33. data/lib/fog/compute/serverlove/requests/shutdown_server.rb +11 -0
  34. data/lib/fog/compute/serverlove/requests/start_server.rb +11 -0
  35. data/lib/fog/compute/serverlove/requests/stop_server.rb +11 -0
  36. data/lib/fog/compute/serverlove/requests/update_image.rb +13 -0
  37. data/lib/fog/compute/serverlove/requests/update_server.rb +13 -0
  38. data/lib/fog/serverlove.rb +15 -0
  39. data/lib/fog/serverlove/compute.rb +0 -0
  40. data/lib/fog/serverlove/version.rb +5 -0
  41. data/spec/minitest_helper.rb +31 -0
  42. data/tests/compute/password_generator_tests.rb +15 -0
  43. data/tests/compute/requests/image_tests.rb +59 -0
  44. data/tests/compute/requests/server_tests.rb +92 -0
  45. data/tests/helper.rb +34 -0
  46. data/tests/helpers/collection_helper.rb +97 -0
  47. data/tests/helpers/flavors_helper.rb +32 -0
  48. data/tests/helpers/formats_helper.rb +98 -0
  49. data/tests/helpers/formats_helper_tests.rb +110 -0
  50. data/tests/helpers/model_helper.rb +29 -0
  51. data/tests/helpers/responds_to_helper.rb +11 -0
  52. data/tests/helpers/schema_validator_tests.rb +107 -0
  53. data/tests/helpers/server_helper.rb +25 -0
  54. data/tests/helpers/servers_helper.rb +10 -0
  55. data/tests/helpers/succeeds_helper.rb +9 -0
  56. metadata +227 -0
@@ -0,0 +1,92 @@
1
+ Shindo.tests('Fog::Compute[:serverlove] | server requests', ['serverlove']) do
2
+ @server_format = {
3
+ 'server' => String,
4
+ 'name' => String,
5
+ 'user' => String,
6
+ 'status' => String,
7
+ 'started' => Fog::Nullable::String,
8
+ 'cpu' => Integer,
9
+ 'mem' => Integer,
10
+ 'smp' => Fog::Nullable::String,
11
+ 'persistent' => Fog::Nullable::String,
12
+ 'vnc' => Fog::Nullable::String,
13
+ 'vnc:password' => Fog::Nullable::String,
14
+ 'nic:0:dhcp' => String,
15
+ 'nic:0:model' => String
16
+ }
17
+
18
+ tests('success') do
19
+ attributes = { 'name' => 'Test', 'cpu' => '1000', 'mem' => '1000', 'persistent' => 'true' }
20
+
21
+ tests("#create_server").formats(@server_format) do
22
+ pending
23
+ @server = Fog::Compute[:serverlove].create_server(Fog::Compute::Serverlove::Server.defaults.merge(attributes)).body
24
+ end
25
+
26
+ tests("#list_servers").succeeds do
27
+ pending
28
+ Fog::Compute[:serverlove].servers
29
+ end
30
+
31
+ tests("#update_server").returns(true) do
32
+ pending
33
+ @server['name'] = "Diff"
34
+ Fog::Compute[:serverlove].update_server(@server['server'], { :name => @server['name']})
35
+ Fog::Compute[:serverlove].servers.get(@server['server']).name == "Diff"
36
+ end
37
+
38
+ tests("assigns drive to server").succeeds do
39
+ pending
40
+ @image = Fog::Compute[:serverlove].create_image('name' => 'Test', 'size' => '24234567890').body
41
+ # Load debian
42
+ Fog::Compute[:serverlove].load_standard_image(@image['drive'], 'aca2fa0b-40bc-4e06-ad99-f1467690d5de')
43
+ Fog::Compute[:serverlove].update_server(@server['server'], { 'ide:0:0' => @image['drive'], 'boot' => 'ide:0:0'})
44
+ end
45
+
46
+ tests("waits for imaging...").returns(true) do
47
+ pending
48
+ while(percent_complete = Fog::Compute[:serverlove].images.get(@image['drive']).imaging)
49
+ sleep(1)
50
+ STDERR.print "#{percent_complete} "
51
+ break if percent_complete.include?("100")
52
+ end
53
+ STDERR.print "100% "
54
+ true
55
+ end
56
+
57
+ tests("#start_server").returns(true) do
58
+ pending
59
+ Fog::Compute[:serverlove].start_server(@server['server'])
60
+ Fog::Compute[:serverlove].servers.get(@server['server']).status == "active"
61
+ end
62
+
63
+ tests("#reset_server").returns(true) do
64
+ pending
65
+ Fog::Compute[:serverlove].reset_server(@server['server'])
66
+ Fog::Compute[:serverlove].servers.get(@server['server']).status == "active"
67
+ end
68
+
69
+ tests("#shutdown_server").succeeds do
70
+ pending
71
+ Fog::Compute[:serverlove].shutdown_server(@server['server'])
72
+ # Can't guarantee the OS will honour this command so don't test status
73
+ end
74
+
75
+ tests("#stop_server").returns(true) do
76
+ pending
77
+ Fog::Compute[:serverlove].start_server(@server['server'])
78
+ Fog::Compute[:serverlove].stop_server(@server['server'])
79
+ Fog::Compute[:serverlove].servers.get(@server['server']).status == "stopped"
80
+ end
81
+
82
+ tests("#destroy_server").succeeds do
83
+ pending
84
+ Fog::Compute[:serverlove].destroy_server(@server['server'])
85
+ end
86
+
87
+ tests("destroy test drive").succeeds do
88
+ pending
89
+ Fog::Compute[:serverlove].destroy_image(@image['drive'])
90
+ end
91
+ end
92
+ end
data/tests/helper.rb ADDED
@@ -0,0 +1,34 @@
1
+ require 'excon'
2
+
3
+ if ENV['COVERAGE']
4
+ require 'coveralls'
5
+ require 'simplecov'
6
+
7
+ SimpleCov.start do
8
+ add_filter '/spec/'
9
+ add_filter '/test/'
10
+ end
11
+ end
12
+
13
+ require File.expand_path(File.join(File.dirname(__FILE__), '../lib/fog/serverlove'))
14
+
15
+ Coveralls.wear! if ENV['COVERAGE']
16
+
17
+ Excon.defaults.merge!(:debug_request => true, :debug_response => true)
18
+
19
+ # This overrides the default 600 seconds timeout during live test runs
20
+ if Fog.mocking?
21
+ FOG_TESTING_TIMEOUT = ENV['FOG_TEST_TIMEOUT'] || 2000
22
+ Fog.timeout = 2000
23
+ Fog::Logger.warning "Setting default fog timeout to #{Fog.timeout} seconds"
24
+ else
25
+ FOG_TESTING_TIMEOUT = Fog.timeout
26
+ end
27
+
28
+ def lorem_file
29
+ File.open(File.dirname(__FILE__) + '/lorem.txt', 'r')
30
+ end
31
+
32
+ def array_differences(array_a, array_b)
33
+ (array_a - array_b) | (array_b - array_a)
34
+ end
@@ -0,0 +1,97 @@
1
+ def collection_tests(collection, params = {}, mocks_implemented = true)
2
+ tests('success') do
3
+
4
+ tests("#new(#{params.inspect})").succeeds do
5
+ pending if Fog.mocking? && !mocks_implemented
6
+ collection.new(params)
7
+ end
8
+
9
+ tests("#create(#{params.inspect})").succeeds do
10
+ pending if Fog.mocking? && !mocks_implemented
11
+ @instance = collection.create(params)
12
+ end
13
+ # FIXME: work around for timing issue on AWS describe_instances mocks
14
+
15
+ if Fog.mocking? && @instance.respond_to?(:ready?)
16
+ @instance.wait_for { ready? }
17
+ end
18
+
19
+ tests("#all").succeeds do
20
+ pending if Fog.mocking? && !mocks_implemented
21
+ collection.all
22
+ end
23
+
24
+ if !Fog.mocking? || mocks_implemented
25
+ @identity = @instance.identity
26
+ end
27
+
28
+ tests("#get(#{@identity})").succeeds do
29
+ pending if Fog.mocking? && !mocks_implemented
30
+ collection.get(@identity)
31
+ end
32
+
33
+ tests('Enumerable') do
34
+ pending if Fog.mocking? && !mocks_implemented
35
+
36
+ methods = [
37
+ 'all?', 'any?', 'find', 'detect', 'collect', 'map',
38
+ 'find_index', 'flat_map', 'collect_concat', 'group_by',
39
+ 'none?', 'one?'
40
+ ]
41
+
42
+ # JRuby 1.7.5+ issue causes a SystemStackError: stack level too deep
43
+ # https://github.com/jruby/jruby/issues/1265
44
+ if RUBY_PLATFORM == "java" and JRUBY_VERSION =~ /1\.7\.[5-8]/
45
+ methods.delete('all?')
46
+ end
47
+
48
+ methods.each do |enum_method|
49
+ if collection.respond_to?(enum_method)
50
+ tests("##{enum_method}").succeeds do
51
+ block_called = false
52
+ collection.send(enum_method) {|x| block_called = true }
53
+ block_called
54
+ end
55
+ end
56
+ end
57
+
58
+ [
59
+ 'max_by','min_by'
60
+ ].each do |enum_method|
61
+ if collection.respond_to?(enum_method)
62
+ tests("##{enum_method}").succeeds do
63
+ block_called = false
64
+ collection.send(enum_method) {|x| block_called = true; 0 }
65
+ block_called
66
+ end
67
+ end
68
+
69
+ end
70
+
71
+ end
72
+
73
+ if block_given?
74
+ yield(@instance)
75
+ end
76
+
77
+ if !Fog.mocking? || mocks_implemented
78
+ @instance.destroy
79
+ end
80
+ end
81
+
82
+ tests('failure') do
83
+
84
+ if !Fog.mocking? || mocks_implemented
85
+ @identity = @identity.to_s
86
+ @identity = @identity.gsub(/[a-zA-Z]/) { Fog::Mock.random_letters(1) }
87
+ @identity = @identity.gsub(/\d/) { Fog::Mock.random_numbers(1) }
88
+ @identity
89
+ end
90
+
91
+ tests("#get('#{@identity}')").returns(nil) do
92
+ pending if Fog.mocking? && !mocks_implemented
93
+ collection.get(@identity)
94
+ end
95
+
96
+ end
97
+ end
@@ -0,0 +1,32 @@
1
+ def flavors_tests(connection, params = {}, mocks_implemented = true)
2
+ tests('success') do
3
+
4
+ tests("#all").succeeds do
5
+ pending if Fog.mocking? && !mocks_implemented
6
+ connection.flavors.all
7
+ end
8
+
9
+ if !Fog.mocking? || mocks_implemented
10
+ @identity = connection.flavors.first.identity
11
+ end
12
+
13
+ tests("#get('#{@identity}')").succeeds do
14
+ pending if Fog.mocking? && !mocks_implemented
15
+ connection.flavors.get(@identity)
16
+ end
17
+
18
+ end
19
+
20
+ tests('failure') do
21
+
22
+ if !Fog.mocking? || mocks_implemented
23
+ invalid_flavor_identity = connection.flavors.first.identity.to_s.gsub(/\w/, '0')
24
+ end
25
+
26
+ tests("#get('#{invalid_flavor_identity}')").returns(nil) do
27
+ pending if Fog.mocking? && !mocks_implemented
28
+ connection.flavors.get(invalid_flavor_identity)
29
+ end
30
+
31
+ end
32
+ end
@@ -0,0 +1,98 @@
1
+ require "fog/schema/data_validator"
2
+
3
+ # format related hackery
4
+ # allows both true.is_a?(Fog::Boolean) and false.is_a?(Fog::Boolean)
5
+ # allows both nil.is_a?(Fog::Nullable::String) and ''.is_a?(Fog::Nullable::String)
6
+ module Fog
7
+ module Boolean; end
8
+ module Nullable
9
+ module Boolean; end
10
+ module Integer; end
11
+ module String; end
12
+ module Time; end
13
+ module Float; end
14
+ module Hash; end
15
+ module Array; end
16
+ end
17
+ end
18
+ [FalseClass, TrueClass].each {|klass| klass.send(:include, Fog::Boolean)}
19
+ [FalseClass, TrueClass, NilClass, Fog::Boolean].each {|klass| klass.send(:include, Fog::Nullable::Boolean)}
20
+ [NilClass, String].each {|klass| klass.send(:include, Fog::Nullable::String)}
21
+ [NilClass, Time].each {|klass| klass.send(:include, Fog::Nullable::Time)}
22
+ [Integer, NilClass].each {|klass| klass.send(:include, Fog::Nullable::Integer)}
23
+ [Float, NilClass].each {|klass| klass.send(:include, Fog::Nullable::Float)}
24
+ [Hash, NilClass].each {|klass| klass.send(:include, Fog::Nullable::Hash)}
25
+ [Array, NilClass].each {|klass| klass.send(:include, Fog::Nullable::Array)}
26
+
27
+ module Shindo
28
+ class Tests
29
+ # Generates a Shindo test that compares a hash schema to the result
30
+ # of the passed in block returning true if they match.
31
+ #
32
+ # The schema that is passed in is a Hash or Array of hashes that
33
+ # have Classes in place of values. When checking the schema the
34
+ # value should match the Class.
35
+ #
36
+ # Strict mode will fail if the data has additional keys. Setting
37
+ # +strict+ to +false+ will allow additional keys to appear.
38
+ #
39
+ # @param [Hash] schema A Hash schema
40
+ # @param [Hash] options Options to change validation rules
41
+ # @option options [Boolean] :allow_extra_keys
42
+ # If +true+ does not fail when keys are in the data that are
43
+ # not specified in the schema. This allows new values to
44
+ # appear in API output without breaking the check.
45
+ # @option options [Boolean] :allow_optional_rules
46
+ # If +true+ does not fail if extra keys are in the schema
47
+ # that do not match the data. Not recommended!
48
+ # @yield Data to check with schema
49
+ #
50
+ # @example Using in a test
51
+ # Shindo.tests("comparing welcome data against schema") do
52
+ # data = {:welcome => "Hello" }
53
+ # data_matches_schema(:welcome => String) { data }
54
+ # end
55
+ #
56
+ # comparing welcome data against schema
57
+ # + data matches schema
58
+ #
59
+ # @example Example schema
60
+ # {
61
+ # "id" => String,
62
+ # "ram" => Integer,
63
+ # "disks" => [
64
+ # {
65
+ # "size" => Float
66
+ # }
67
+ # ],
68
+ # "dns_name" => Fog::Nullable::String,
69
+ # "active" => Fog::Boolean,
70
+ # "created" => DateTime
71
+ # }
72
+ #
73
+ # @return [Boolean]
74
+ def data_matches_schema(schema, options = {})
75
+ test('data matches schema') do
76
+ validator = Fog::Schema::DataValidator.new
77
+ valid = validator.validate(yield, schema, options)
78
+ @message = validator.message unless valid
79
+ valid
80
+ end
81
+ end
82
+
83
+ # @deprecated #formats is deprecated. Use #data_matches_schema instead
84
+ def formats(format, strict = true)
85
+ test('has proper format') do
86
+ if strict
87
+ options = {:allow_extra_keys => false, :allow_optional_rules => true}
88
+ else
89
+ options = {:allow_extra_keys => true, :allow_optional_rules => true}
90
+ end
91
+ validator = Fog::Schema::DataValidator.new
92
+ valid = validator.validate(yield, format, options)
93
+ @message = validator.message unless valid
94
+ valid
95
+ end
96
+ end
97
+ end
98
+ end
@@ -0,0 +1,110 @@
1
+ Shindo.tests('test_helper', 'meta') do
2
+
3
+ tests('comparing welcome data against schema') do
4
+ data = {:welcome => "Hello" }
5
+ data_matches_schema(:welcome => String) { data }
6
+ end
7
+
8
+ tests('#data_matches_schema') do
9
+ tests('when value matches schema expectation') do
10
+ data_matches_schema({"key" => String}) { {"key" => "Value"} }
11
+ end
12
+
13
+ tests('when values within an array all match schema expectation') do
14
+ data_matches_schema({"key" => [Integer]}) { {"key" => [1, 2]} }
15
+ end
16
+
17
+ tests('when nested values match schema expectation') do
18
+ data_matches_schema({"key" => {:nested_key => String}}) { {"key" => {:nested_key => "Value"}} }
19
+ end
20
+
21
+ tests('when collection of values all match schema expectation') do
22
+ data_matches_schema([{"key" => String}]) { [{"key" => "Value"}, {"key" => "Value"}] }
23
+ end
24
+
25
+ tests('when collection is empty although schema covers optional members') do
26
+ data_matches_schema([{"key" => String}], {:allow_optional_rules => true}) { [] }
27
+ end
28
+
29
+ tests('when additional keys are passed and not strict') do
30
+ data_matches_schema({"key" => String}, {:allow_extra_keys => true}) { {"key" => "Value", :extra => "Bonus"} }
31
+ end
32
+
33
+ tests('when value is nil and schema expects NilClass') do
34
+ data_matches_schema({"key" => NilClass}) { {"key" => nil} }
35
+ end
36
+
37
+ tests('when value and schema match as hashes') do
38
+ data_matches_schema({}) { {} }
39
+ end
40
+
41
+ tests('when value and schema match as arrays') do
42
+ data_matches_schema([]) { [] }
43
+ end
44
+
45
+ tests('when value is a Time') do
46
+ data_matches_schema({"time" => Time}) { {"time" => Time.now} }
47
+ end
48
+
49
+ tests('when key is missing but value should be NilClass (#1477)') do
50
+ data_matches_schema({"key" => NilClass}, {:allow_optional_rules => true}) { {} }
51
+ end
52
+
53
+ tests('when key is missing but value is nullable (#1477)') do
54
+ data_matches_schema({"key" => Fog::Nullable::String}, {:allow_optional_rules => true}) { {} }
55
+ end
56
+ end
57
+
58
+ tests('#formats backwards compatible changes') do
59
+
60
+ tests('when value matches schema expectation') do
61
+ formats({"key" => String}) { {"key" => "Value"} }
62
+ end
63
+
64
+ tests('when values within an array all match schema expectation') do
65
+ formats({"key" => [Integer]}) { {"key" => [1, 2]} }
66
+ end
67
+
68
+ tests('when nested values match schema expectation') do
69
+ formats({"key" => {:nested_key => String}}) { {"key" => {:nested_key => "Value"}} }
70
+ end
71
+
72
+ tests('when collection of values all match schema expectation') do
73
+ formats([{"key" => String}]) { [{"key" => "Value"}, {"key" => "Value"}] }
74
+ end
75
+
76
+ tests('when collection is empty although schema covers optional members') do
77
+ formats([{"key" => String}]) { [] }
78
+ end
79
+
80
+ tests('when additional keys are passed and not strict') do
81
+ formats({"key" => String}, false) { {"key" => "Value", :extra => "Bonus"} }
82
+ end
83
+
84
+ tests('when value is nil and schema expects NilClass') do
85
+ formats({"key" => NilClass}) { {"key" => nil} }
86
+ end
87
+
88
+ tests('when value and schema match as hashes') do
89
+ formats({}) { {} }
90
+ end
91
+
92
+ tests('when value and schema match as arrays') do
93
+ formats([]) { [] }
94
+ end
95
+
96
+ tests('when value is a Time') do
97
+ formats({"time" => Time}) { {"time" => Time.now} }
98
+ end
99
+
100
+ tests('when key is missing but value should be NilClass (#1477)') do
101
+ formats({"key" => NilClass}) { {} }
102
+ end
103
+
104
+ tests('when key is missing but value is nullable (#1477)') do
105
+ formats({"key" => Fog::Nullable::String}) { {} }
106
+ end
107
+
108
+ end
109
+
110
+ end