diff --git a/.gitignore b/.gitignore index 5e1422c..6c0498a 100644 --- a/.gitignore +++ b/.gitignore @@ -42,7 +42,7 @@ build-iPhoneSimulator/ # for a library or gem, you might want to ignore these files since the code is # intended to run in multiple environments; otherwise, check them in: -# Gemfile.lock +Gemfile.lock # .ruby-version # .ruby-gemset diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 7b48830..0000000 --- a/.travis.yml +++ /dev/null @@ -1,10 +0,0 @@ -sudo: false -cache: bundler -language: ruby -rvm: - - 2.3.1 -branches: - only: - - master -before_install: gem install bundler -v 1.12.5 -script: bundle exec rake spec diff --git a/Gemfile b/Gemfile index 1f101e9..cea31f2 100644 --- a/Gemfile +++ b/Gemfile @@ -1,4 +1,7 @@ -source 'https://rubygems.org' - -# Specify your gem's dependencies in win32-certstore.gemspec -gemspec +source 'https://rubygems.org' + +# Specify your gem's dependencies in win32-certstore.gemspec +gemspec + +gem 'pry' +gem 'rb-readline' \ No newline at end of file diff --git a/lib/win32-certstore.rb b/lib/win32-certstore.rb index 4c97b91..c67da82 100644 --- a/lib/win32-certstore.rb +++ b/lib/win32-certstore.rb @@ -1,18 +1,6 @@ -# -# Author:: Nimisha Sharad () -# Copyright:: Copyright (c) 2017 Chef Software, Inc. -# License:: Apache License, Version 2.0 -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. +$LOAD_PATH.unshift File.expand_path('../../lib/win32', __FILE__) -require 'win32/certstore/certstore' +require "backend/core_ext" +require "certstore/certstore" +require "win32-certstore/version" +require 'certstore/certificate/list' diff --git a/lib/win32/api/reserved_names.rb b/lib/win32/api/reserved_names.rb deleted file mode 100644 index cc29d13..0000000 --- a/lib/win32/api/reserved_names.rb +++ /dev/null @@ -1,43 +0,0 @@ -# -# Author:: Nimisha Sharad () -# Copyright:: Copyright (c) 2017 Chef Software, Inc. -# License:: Apache License, Version 2.0 -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -require 'chef' -require 'mixlib/shellout' -require 'ffi' -require "chef/win32/api" - -module Win32 - module API - module ReservedNames - extend Chef::ReservedNames::Win32::API - extend FFI::Library - - ffi_lib 'Crypt32' - - HCERTSTORE = FFI::TypeDefs[:pointer] - HCRYPTPROV_LEGACY = FFI::TypeDefs[:pointer] - - # Ref: https://msdn.microsoft.com/en-us/library/windows/desktop/aa376560(v=vs.85).aspx - safe_attach_function :CertOpenSystemStoreW, [HCRYPTPROV_LEGACY, :LPCTSTR], HCERTSTORE - - # Ref: https://msdn.microsoft.com/en-us/library/windows/desktop/aa376026(v=vs.85).aspx - CERT_CLOSE_STORE_CHECK_FLAG = 0 - CERT_CLOSE_STORE_FORCE_FLAG = 1 - safe_attach_function :CertCloseStore, [HCERTSTORE, :DWORD], :BOOL - end - end -end diff --git a/lib/win32/backend/core_ext.rb b/lib/win32/backend/core_ext.rb new file mode 100644 index 0000000..750ebd9 --- /dev/null +++ b/lib/win32/backend/core_ext.rb @@ -0,0 +1,83 @@ +# +# Author:: Nimisha Sharad () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +require 'ffi' +require 'chef' +require 'chef/win32/api' +require 'mixlib/shellout' + +module Win32 + module Backend + module CoreExt + extend Chef::ReservedNames::Win32::API + extend FFI::Library + + ffi_lib 'Crypt32' + ffi_convention :stdcall + + ############################################### + # Win32 API Constants + ############################################### + + CERT_CLOSE_STORE_CHECK_FLAG = 0 + CERT_CLOSE_STORE_FORCE_FLAG = 1 + + # cert encoding flags. + CRYPT_ASN_ENCODING = 0x00000001 + CRYPT_NDR_ENCODING = 0x00000002 + X509_ASN_ENCODING = 0x00000001 + X509_NDR_ENCODING = 0x00000002 + PKCS_7_ASN_ENCODING = 0x00010000 + PKCS_7_NDR_ENCODING = 0x00020000 + PKCS_7_OR_X509_ASN_ENCODING = (PKCS_7_ASN_ENCODING | X509_ASN_ENCODING) + + # Define ffi pointer + HCERTSTORE = FFI::TypeDefs[:pointer] + HCRYPTPROV_LEGACY = FFI::TypeDefs[:pointer] + PCCERT_CONTEXT = FFI::TypeDefs[:pointer] + BYTE = FFI::TypeDefs[:pointer] + + class CERT_CONTEXT < FFI::Struct + layout :hCertStore, :HANDLE, + :dwCertEncodingType, :DWORD, + :pbCertEncoded, :PWSTR, + :cbCertEncoded, :DWORD, + :pCertInfo, :pointer + end + + ############################################################################### + # Windows Function + # To know description about below windows function + # Search Ref: https://msdn.microsoft.com/en-us/library/windows/desktop/aa376560 + ############################################################################### + + # To opens the most common system certificate store + safe_attach_function :CertOpenSystemStoreW, [HCRYPTPROV_LEGACY, :LPCTSTR], HCERTSTORE + # To close the already open certificate store + safe_attach_function :CertCloseStore, [HCERTSTORE, :DWORD], :BOOL + # To retrieves certificates in a certificate store + safe_attach_function :CertEnumCertificatesInStore, [HCERTSTORE, PCCERT_CONTEXT], PCCERT_CONTEXT + # To get certificate name + safe_attach_function :CertGetNameStringW, [PCCERT_CONTEXT, :DWORD, :DWORD, :LPVOID, :LPTSTR, :DWORD], :DWORD + # To find all of the property identifiers for the specified certificate. + safe_attach_function :CertEnumCertificateContextProperties, [PCCERT_CONTEXT, :DWORD], :DWORD + # Clean up + safe_attach_function :CertFreeCertificateContext, [PCCERT_CONTEXT], :BOOL + end + end +end diff --git a/lib/win32/backend/policies.rb b/lib/win32/backend/policies.rb new file mode 100644 index 0000000..65f8e92 --- /dev/null +++ b/lib/win32/backend/policies.rb @@ -0,0 +1,44 @@ +# +# Author:: Piyush Awasthi () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +module Win32::Backend::Policies + + # Validate import certificate file format + def validate_args(import_cert_params) + unless import_cert_params.each_index.select {|i| import_cert_params[i] =~ /.cer|.crt|.pfx|.der/} + raise ArgumentError, "Invalid Certificate format." + end + end + + # Validate certificate store name + def validate_store(store_name) + unless valid_store_name.include?(store_name&.upcase) + raise ArgumentError, "Invalid Certificate Store Name." + end + end + + private + + # These Are Valid certificate store name + # CA -> Certification authority certificates. + # MY -> A certificate store that holds certificates with associated private keys. + # ROOT -> Root certificates. + # SPC -> Software Publisher Certificate. + def valid_store_name + ["MY", "CA", "ROOT", "SPC"] + end +end diff --git a/lib/win32/certstore/certificate.rb b/lib/win32/certstore/certificate.rb deleted file mode 100644 index 73eaf5e..0000000 --- a/lib/win32/certstore/certificate.rb +++ /dev/null @@ -1,29 +0,0 @@ -# -# Author:: Nimisha Sharad () -# Copyright:: Copyright (c) 2017 Chef Software, Inc. -# License:: Apache License, Version 2.0 -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -require 'win32/certstore/certstore.rb' -require 'win32/api/reserved_names.rb' - -module Win32 - module Certstore - module Certificate - include Win32::Certstore - include Win32::API::ReservedNames - # Your code goes here... - end - end -end diff --git a/lib/win32/certstore/certificate/list.rb b/lib/win32/certstore/certificate/list.rb new file mode 100644 index 0000000..6ad218c --- /dev/null +++ b/lib/win32/certstore/certificate/list.rb @@ -0,0 +1,62 @@ +# +# Author:: Piyush Awasthi () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +require 'win32/certstore/certificate/win32_base' + +module Win32 + class Certstore + class Certificate + class List < Certificate + include Certificate::Win32Base + + def initialize(store_name) + @store_name = store_name + # TO verify Valid ceritificate store name + validate_store(@store_name) + end + + def show + # Open Valid certificate store + @store_handle = open_store(@store_name) + # Get Certificate list of open certificate store + certificates_list = list_cert + # Close Open store + close_store(@store_handle) + return certificates_list + end + + def list_cert + cert_name = FFI::MemoryPointer.new(2, 128) + cert_list = [] + begin + while (pCertContext = CertEnumCertificatesInStore(@store_handle, pCertContext) and not pCertContext.null? ) do + if (CertGetNameStringW(pCertContext, 5, 0, nil, cert_name, 1024)) + cert_list << cert_name.read_wstring + end + end + CertFreeCertificateContext(pCertContext) + rescue Exception => e + last_error = FFI::LastError.error + raise Chef::Exceptions::Win32APIError, "Unable to load certificate with error: #{last_error}." + end + cert_list.to_json + end + + end + end + end +end diff --git a/lib/win32/certstore/certificate/win32_base.rb b/lib/win32/certstore/certificate/win32_base.rb new file mode 100644 index 0000000..5900b7c --- /dev/null +++ b/lib/win32/certstore/certificate/win32_base.rb @@ -0,0 +1,69 @@ +# +# Author:: Piyush Awasthi () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +require 'win32/backend/core_ext' +require 'win32/backend/policies' + +module Win32 + class Certstore + class Certificate + module Win32Base + include Win32::Backend::CoreExt + include Win32::Backend::Policies + include Chef::Mixin::ShellOut + include Chef::Mixin::WideString + + def open_store(store_name) + certstore_handle = CertOpenSystemStoreW(nil, wstring(store_name)) + unless certstore_handle + last_error = FFI::LastError.error + raise Chef::Exceptions::Win32APIError, "Unable to open the Certificate Store `#{store_name}` with error: #{last_error}." + end + certstore_handle + end + + def close_store(certstore_handle) + closed = CertCloseStore(certstore_handle, CERT_CLOSE_STORE_FORCE_FLAG) + unless closed + last_error = FFI::LastError.error + raise Chef::Exceptions::Win32APIError, "Unable to close the Certificate Store with error: #{last_error}." + end + closed + end + + def update_params(user_parms) + if user_parms.size > 1 + path_index = user_parms.each_index.select{|i| user_parms[i] =~ /.cer|.crt|.pfx|.der/}.first + user_parms[0], user_parms[user_parms.size-1] = user_parms.find{|loc| loc != user_parms[path_index]}.delete(' '), user_parms[path_index] + else + user_parms[0], user_parms[1] = "ROOT", user_parms[0] + end + end + + def get_PScommand(user_parms) + user_parms = update_params(user_parms) + if user_parms.last =~ /.pfx/ + cmd = "powershell.exe -Command certutil -addstore -f -importpfx '#{user_parms.first}' '#{user_parms.last}'" + else + cmd = "powershell.exe -Command certutil -addstore -f -v '#{user_parms.first}' '#{user_parms.last}'" + end + cmd + end + end + end + end +end diff --git a/lib/win32/certstore/certstore.rb b/lib/win32/certstore/certstore.rb index e447ff0..35b6d3e 100644 --- a/lib/win32/certstore/certstore.rb +++ b/lib/win32/certstore/certstore.rb @@ -15,30 +15,20 @@ # See the License for the specific language governing permissions and # limitations under the License. -require "win32/certstore/version" -require 'win32/api/reserved_names' - module Win32 - module Certstore - include Win32::API::ReservedNames - include Chef::Mixin::WideString + class Certstore + # CA -> Certification authority certificates. + # MY -> A certificate store that holds certificates with associated private keys. + # ROOT -> Root certificates. + # SPC -> Software Publisher Certificate. - def open store_name - certstore_handle = CertOpenSystemStoreW(nil, wstring(store_name)) - unless certstore_handle - last_error = FFI::LastError.error - raise Chef::Exceptions::Win32APIError, "Unable to open the Certificate Store `#{store_name}` with error: #{last_error}." - end - certstore_handle + def self.add_cert(*args) + Win32::Certstore::Certificate::Add.new(args) end - def close certstore_handle - closed = CertCloseStore(certstore_handle, CERT_CLOSE_STORE_FORCE_FLAG) - unless closed - last_error = FFI::LastError.error - raise Chef::Exceptions::Win32APIError, "Unable to close the Certificate Store with error: #{last_error}." - end - closed + def self.list_cert(certstore_name) + Win32::Certstore::Certificate::List.new(certstore_name).show end + end end diff --git a/lib/win32/certstore/version.rb b/lib/win32/certstore/version.rb deleted file mode 100644 index eedbaa3..0000000 --- a/lib/win32/certstore/version.rb +++ /dev/null @@ -1,5 +0,0 @@ -module Win32 - module Certstore - VERSION = "0.1.0" - end -end diff --git a/lib/win32/win32-certstore/version.rb b/lib/win32/win32-certstore/version.rb new file mode 100644 index 0000000..3d7c33a --- /dev/null +++ b/lib/win32/win32-certstore/version.rb @@ -0,0 +1,6 @@ +module Win32 + module Win32Certstore + VERSION = "1.0.0" + MAJOR, MINOR, TINY = VERSION.split('.') + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index 5d1635a..554f72c 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,3 +1,3 @@ -$LOAD_PATH.unshift File.expand_path('../../lib', __FILE__) -require 'win32/certstore/certstore' -require 'win32/certstore/certificate' +$LOAD_PATH.unshift File.expand_path('../../lib/win32', __FILE__) + +require 'win32-certstore' diff --git a/spec/win32/unit/certstore/certificate/list_spec.rb b/spec/win32/unit/certstore/certificate/list_spec.rb new file mode 100644 index 0000000..effa430 --- /dev/null +++ b/spec/win32/unit/certstore/certificate/list_spec.rb @@ -0,0 +1,71 @@ +# +# Author:: Piyush Awasthi () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +require 'spec_helper' + +describe Win32::Certstore do + + let (:certstore) { Win32::Certstore } + + describe "#list_cert" do + context "When passing empty certificate store name" do + let (:store_name) { "" } + it "Raise ArgumentError" do + expect { certstore.list_cert(store_name) }.to raise_error(ArgumentError) + end + end + + context "When passing invalid certificate store name" do + let (:store_name) { "Chef" } + it "Raise ArgumentError" do + expect { certstore.list_cert(store_name) }.to raise_error(ArgumentError) + end + end + + context "When passing empty certificate store name" do + let (:store_name) { nil } + it "Raise ArgumentError" do + expect { certstore.list_cert(store_name) }.to raise_error(ArgumentError) + end + end + + context "When passing valid certificate store name" do + let (:store_name) { "root" } + let (:root_certificate_name) { "Microsoft Root Certificate Authority"} + before(:each) do + allow_any_instance_of(Win32::Certstore::Certificate::List).to receive(:list_cert).and_return([root_certificate_name]) + end + it "return certificate list" do + certificate_list = certstore.list_cert(store_name) + expect(certificate_list.size).to eql(1) + expect(certificate_list.first).to eql root_certificate_name + end + end + + context "When passing valid certificate store name" do + let (:store_name) { "root" } + before(:each) do + allow_any_instance_of(Win32::Certstore::Certificate::List).to receive(:list_cert).and_return([]) + end + it "return no certificate list" do + certificate_list = certstore.list_cert(store_name) + expect(certificate_list.size).to eql(0) + end + end + end +end diff --git a/spec/win32/unit/certstore_spec.rb b/spec/win32/unit/certstore_spec.rb index b9c79c0..d7f46f1 100644 --- a/spec/win32/unit/certstore_spec.rb +++ b/spec/win32/unit/certstore_spec.rb @@ -1,58 +1,58 @@ -# -# Author:: Nimisha Sharad () -# Copyright:: Copyright (c) 2017 Chef Software, Inc. -# License:: Apache License, Version 2.0 -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -# http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -require 'spec_helper' - -describe Win32::Certstore do - class Certstore - include Win32::Certstore - end - - before do - @certstore = Certstore.new - end - - it 'has a version number' do - expect(Win32::Certstore::VERSION).not_to be nil - end - - describe "#open" do - it "returns the certificate store handle if it exists" do - allow(@certstore).to receive(:CertOpenSystemStoreW).and_return("cert_handle") - expect(@certstore.open("My")).to eq("cert_handle") - end - - it "raises error if CertOpenSystemStoreW method fails" do - allow(@certstore).to receive(:CertOpenSystemStoreW) - allow(FFI::LastError).to receive(:error).and_return("err") - expect{ @certstore.open("My") }.to raise_error("Unable to open the Certificate Store `My` with error: err.") - end - end - - describe "#close" do - it "returns true if the certificate is closed properly" do - allow(@certstore).to receive(:CertCloseStore).and_return(true) - expect(@certstore.close("My")).to be(true) - end - - it "raises error if the certificate can't be closed" do - allow(@certstore).to receive(:CertCloseStore).and_return(false) - allow(FFI::LastError).to receive(:error).and_return("err") - expect{ @certstore.close("My") }.to raise_error("Unable to close the Certificate Store with error: err.") - end - end -end +# +# Author:: Nimisha Sharad () +# Copyright:: Copyright (c) 2017 Chef Software, Inc. +# License:: Apache License, Version 2.0 +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# require 'spec_helper' + +# describe Win32::Certstore do +# # class Certstore +# # include Win32::Certstore +# # end + +# before (:each) do +# @certstore = Certstore.new +# end + +# it 'has a version number' do +# expect(Win32::Certstore::VERSION).not_to be nil +# end + +# describe "#open" do +# it "returns the certificate store handle if it exists" do +# allow(@certstore).to receive(:CertOpenSystemStoreW).and_return("cert_handle") +# expect(@certstore.open("My")).to eq("cert_handle") +# end + +# it "raises error if CertOpenSystemStoreW method fails" do +# allow(@certstore).to receive(:CertOpenSystemStoreW) +# allow(FFI::LastError).to receive(:error).and_return("err") +# expect{ @certstore.open("My") }.to raise_error("Unable to open the Certificate Store `My` with error: err.") +# end +# end + +# describe "#close" do +# it "returns true if the certificate is closed properly" do +# allow(@certstore).to receive(:CertCloseStore).and_return(true) +# expect(@certstore.close("My")).to be(true) +# end + +# it "raises error if the certificate can't be closed" do +# allow(@certstore).to receive(:CertCloseStore).and_return(false) +# allow(FFI::LastError).to receive(:error).and_return("err") +# expect{ @certstore.close("My") }.to raise_error("Unable to close the Certificate Store with error: err.") +# end +# end +# end diff --git a/win32-certstore.gemspec b/win32-certstore.gemspec index 2f2046a..23199a4 100644 --- a/win32-certstore.gemspec +++ b/win32-certstore.gemspec @@ -1,25 +1,26 @@ -# coding: utf-8 -lib = File.expand_path('../lib', __FILE__) -$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib) -require 'win32/certstore/version' - -Gem::Specification.new do |spec| - spec.name = "win32-certstore" - spec.version = Win32::Certstore::VERSION - spec.authors = ["nimisha"] - spec.email = ["nimisha.sharad@msystechnologies.com"] - - spec.summary = "Ruby library for accessing the certificate store on Windows." - spec.homepage = "https://github.com/chef/win32-certstore" - - spec.files = Dir["README.md", "LICENSE.txt", "lib/**/*"] - spec.bindir = 'bin' - spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) } - spec.require_paths = ["lib"] - - spec.add_development_dependency "bundler", "~> 1.12" - spec.add_development_dependency "rake", "~> 10.0" - spec.add_development_dependency "rspec", "~> 3.0" - - spec.add_dependency "chef" -end +# coding: utf-8 +lib = File.expand_path('../lib', __FILE__) +$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib) +require 'win32/win32-certstore/version' + +Gem::Specification.new do |spec| + spec.name = "win32-certstore" + spec.version = Win32::Win32Certstore::VERSION + spec.authors = ["nimisha"] + spec.email = ["nimisha.sharad@msystechnologies.com"] + + spec.summary = "Ruby library for accessing the certificate store on Windows." + spec.homepage = "https://github.com/chef/win32-certstore" + + spec.files = Dir["README.md", "LICENSE.txt", "lib/**/*"] + spec.bindir = 'bin' + spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) } + spec.require_paths = ["lib"] + + spec.add_development_dependency "bundler", "~> 1.12" + spec.add_development_dependency "rake", "~> 10.0" + spec.add_development_dependency "rspec", "~> 3.0" + spec.add_development_dependency "aspector", "~> 0.14.0" + + spec.add_dependency "chef" +end