require 'httpclient' require 'nokogiri' require 'yaml' require 'json' require 'jsonpath' module NexusCli class OSSRemote def initialize(overrides, ssl_verify=true) @configuration = Configuration::parse(overrides) @ssl_verify = ssl_verify end def configuration return @configuration if @configuration end def nexus client = HTTPClient.new client.send_timeout = 600 client.receive_timeout = 600 # https://github.com/nahi/httpclient/issues/63 client.set_auth(nil, configuration['username'], configuration['password']) client.www_auth.basic_auth.challenge(configuration['url']) client.ssl_config.verify_mode = OpenSSL::SSL::VERIFY_NONE unless @ssl_verify return client end def nexus_url(url) File.join(configuration['url'], url) end def status response = nexus.get(nexus_url("service/local/status")) case response.status when 200 doc = Nokogiri::XML(response.content).xpath("/status/data") data = Hash.new data['app_name'] = doc.xpath("appName")[0].text data['version'] = doc.xpath("version")[0].text data['edition_long'] = doc.xpath("editionLong")[0].text data['state'] = doc.xpath("state")[0].text data['started_at'] = doc.xpath("startedAt")[0].text data['base_url'] = doc.xpath("baseUrl")[0].text return data when 401 raise PermissionsException when 503 raise CouldNotConnectToNexusException else raise UnexpectedStatusCodeException.new(response.status) end end def running_nexus_pro? status['edition_long'] == "Professional" end def pull_artifact(artifact, destination=nil) group_id, artifact_id, version, extension = parse_artifact_string(artifact) version = Nokogiri::XML(get_artifact_info(artifact)).xpath("//version").first.content() if version.casecmp("latest") destination = File.join(File.expand_path(destination || "."), "#{artifact_id}-#{version}.#{extension}") response = nexus.get(nexus_url("service/local/artifact/maven/redirect"), :query => {:g => group_id, :a => artifact_id, :v => version, :e => extension, :r => configuration['repository']}) case response.status when 301 # Follow redirect and stream in chunks. artifact_file = File.open(destination, "wb") do |io| nexus.get(response.content.gsub(/If you are not automatically redirected use this url: /, "")) do |chunk| io.write(chunk) end end when 404 raise ArtifactNotFoundException else raise UnexpectedStatusCodeException.new(response.status) end File.expand_path(destination) end def push_artifact(artifact, file) group_id, artifact_id, version, extension = parse_artifact_string(artifact) response = nexus.post(nexus_url("service/local/artifact/maven/content"), {:hasPom => false, :g => group_id, :a => artifact_id, :v => version, :e => extension, :p => extension, :r => configuration['repository'], :file => File.open(file)}) case response.status when 201 return true when 400 raise BadUploadRequestException when 401 raise PermissionsException when 403 raise PermissionsException when 404 raise NexusHTTP404.new(response.content) else raise UnexpectedStatusCodeException.new(response.status) end end def delete_artifact(artifact) group_id, artifact_id, version = parse_artifact_string(artifact) response = nexus.delete(nexus_url("content/repositories/#{configuration['repository']}/#{group_id.gsub(".", "/")}/#{artifact_id.gsub(".", "/")}/#{version}")) case response.status when 204 return true else raise UnexpectedStatusCodeException.new(response.status) end end def get_artifact_info(artifact) group_id, artifact_id, version, extension = parse_artifact_string(artifact) response = nexus.get(nexus_url("service/local/artifact/maven/resolve"), :query => {:g => group_id, :a => artifact_id, :v => version, :e => extension, :r => configuration['repository']}) case response.status when 200 return response.content when 404 raise ArtifactNotFoundException when 503 raise CouldNotConnectToNexusException else raise UnexpectedStatusCodeException.new(response.status) end end def search_for_artifacts(artifact) group_id, artifact_id = artifact.split(":") response = nexus.get(nexus_url("service/local/data_index"), :query => {:g => group_id, :a => artifact_id}) case response.status when 200 doc = Nokogiri::XML(response.content) return format_search_results(doc, group_id, artifact_id) else raise UnexpectedStatusCodeException.new(response.status) end end def get_global_settings json = get_global_settings_json pretty_json = JSON.pretty_generate(JSON.parse(json)) Dir.mkdir(File.expand_path("~/.nexus")) unless Dir.exists?(File.expand_path("~/.nexus")) destination = File.join(File.expand_path("~/.nexus"), "global_settings.json") artifact_file = File.open(destination, 'wb') do |file| file.write(pretty_json) end end def get_global_settings_json response = nexus.get(nexus_url("service/local/global_settings/current"), :header => DEFAULT_ACCEPT_HEADER) case response.status when 200 return response.content else raise UnexpectedStatusCodeException.new(response.status) end end def upload_global_settings(json=nil) global_settings = nil if json == nil global_settings = File.read(File.join(File.expand_path("~/.nexus"), "global_settings.json")) else global_settings = json end response = nexus.put(nexus_url("service/local/global_settings/current"), :body => global_settings, :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 204 return true when 400 raise BadSettingsException.new(response.content) end end def reset_global_settings response = nexus.get(nexus_url("service/local/global_settings/default"), :header => DEFAULT_ACCEPT_HEADER) case response.status when 200 default_json = response.content else raise UnexpectedStatusCodeException.new(response.status) end response = nexus.put(nexus_url("service/local/global_settings/current"), :body => default_json, :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 204 return true else raise UnexpectedStatusCodeException.new(response.status) end end def create_repository(name, proxy, url) json = if proxy create_proxy_repository_json(name, url) else create_hosted_repository_json(name) end response = nexus.post(nexus_url("service/local/repositories"), :body => json, :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 201 return true when 400 raise CreateRepsitoryException.new(response.content) else raise UnexpectedStatusCodeException.new(response.status) end end def delete_repository(name) response = nexus.delete(nexus_url("service/local/repositories/#{sanitize_for_id(name)}")) case response.status when 204 return true when 404 raise RepositoryDoesNotExistException else raise UnexpectedStatusCodeException.new(response.status) end end def get_repository_info(name) response = nexus.get(nexus_url("service/local/repositories/#{sanitize_for_id(name)}")) case response.status when 200 return response.content when 404 raise RepositoryNotFoundException when 503 raise CouldNotConnectToNexusException else raise UnexpectedStatusCodeException.new(response.status) end end def get_users response = nexus.get(nexus_url("service/local/users")) case response.status when 200 return response.content else raise UnexpectedStatusCodeException.new(response.status) end end def create_user(params) response = nexus.post(nexus_url("service/local/users"), :body => create_user_json(params), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 201 return true when 400 raise CreateUserException.new(response.content) else raise UnexpectedStatusCodeException.new(reponse.code) end end def update_user(params) params[:roles] = [] if params[:roles] == [""] user_json = get_user(params[:userId]) modified_json = JsonPath.for(user_json) params.each do |key, value| modified_json.gsub!("$..#{key}"){|v| value} unless key == "userId" || value.blank? end response = nexus.put(nexus_url("service/local/users/#{params[:userId]}"), :body => JSON.dump(modified_json.to_hash), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 200 return true when 400 raise UpdateUserException.new(response.content) else raise UnexpectedStatusCodeException.new(response.status) end end def get_user(user) response = nexus.get(nexus_url("service/local/users/#{user}"), :header => DEFAULT_ACCEPT_HEADER) case response.status when 200 return JSON.parse(response.content) when 404 raise UserNotFoundException.new(user) else raise UnexpectedStatusCodeException.new(response.status) end end def change_password(params) response = nexus.post(nexus_url("service/local/users_changepw"), :body => create_change_password_json(params), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 202 return true when 400 raise InvalidCredentialsException else raise UnexpectedStatusCodeException.new(response.status) end end def delete_user(user_id) response = nexus.delete(nexus_url("service/local/users/#{user_id}")) case response.status when 204 return true when 404 raise UserNotFoundException.new(user_id) else raise UnexpectedStatusCodeException.new(response.status) end end def get_logging_info response = nexus.get(nexus_url("service/local/log/config"), :header => DEFAULT_ACCEPT_HEADER) case response.status when 200 return response.content else raise UnexpectedStatusCodeException.new(response.status) end end def set_logger_level(level) raise InvalidLoggingLevelException unless ["INFO", "DEBUG", "ERROR"].include?(level.upcase) response = nexus.put(nexus_url("service/local/log/config"), :body => create_logger_level_json(level), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 200 return true else raise UnexpectedStatusCodeException.new(response.status) end end def create_group_repository(name) response = nexus.post(nexus_url("service/local/repo_groups"), :body => create_group_repository_json(name), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 201 return true when 400 raise CreateRepsitoryException.new(response.content) else raise UnexpectedStatusCodeException.new(response.status) end end def get_group_repository(group_id) response = nexus.get(nexus_url("service/local/repo_groups/#{sanitize_for_id(group_id)}"), :header => DEFAULT_ACCEPT_HEADER) case response.status when 200 return response.content when 404 raise RepositoryNotFoundException else raise UnexpectedStatusCodeException.new(response.status) end end def repository_in_group?(group_id, repository_to_check) group_repository = JSON.parse(get_group_repository(group_id)) repositories_in_group = group_repository["data"]["repositories"] repositories_in_group.find{|repository| repository["id"] == sanitize_for_id(repository_to_check)} end def add_to_group_repository(group_id, repository_to_add_id) raise RepositoryInGroupException if repository_in_group?(group_id, repository_to_add_id) response = nexus.put(nexus_url("service/local/repo_groups/#{sanitize_for_id(group_id)}"), :body => create_add_to_group_repository_json(group_id, repository_to_add_id), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 200 return true when 400 raise RepositoryNotFoundException else raise UnexpectedStatusCodeException.new(response.status) end end def remove_from_group_repository(group_id, repository_to_remove_id) raise RepositoryNotInGroupException unless repository_in_group?(group_id, repository_to_remove_id) response = nexus.put(nexus_url("service/local/repo_groups/#{sanitize_for_id(group_id)}"), :body => create_remove_from_group_repository_json(group_id, repository_to_remove_id), :header => DEFAULT_CONTENT_TYPE_HEADER) case response.status when 200 return true else raise UnexpectedStatusCodeException.new(response.status) end end def delete_group_repository(group_id) response = nexus.delete(nexus_url("service/local/repo_groups/#{sanitize_for_id(group_id)}")) case response.status when 204 return true when 404 raise RepositoryNotFoundException else raise UnexpectedStatusCodeException.new(response.status) end end def transfer_artifact(artifact, from_repository, to_repository) do_transfer_artifact(artifact, from_repository, to_repository) end private def sanitize_for_id(unsanitized_string) unsanitized_string.gsub(" ", "_").downcase end def do_transfer_artifact(artifact, from_repository, to_repository) Dir.mktmpdir do |temp_dir| configuration["repository"] = sanitize_for_id(from_repository) artifact_file = pull_artifact(artifact, temp_dir) configuration["repository"] = sanitize_for_id(to_repository) push_artifact(artifact, artifact_file) end end def format_search_results(doc, group_id, artifact_id) versions = doc.xpath("//version").inject([]) {|array,node| array << "#{node.content()}"} indent_size = versions.max{|a,b| a.length <=> b.length}.size+4 formated_results = ['Found Versions:'] versions.inject(formated_results) do |array,version| temp_version = version + ":" array << "#{temp_version.ljust(indent_size)} `nexus-cli pull #{group_id}:#{artifact_id}:#{version}:tgz`" end end def parse_artifact_string(artifact) split_artifact = artifact.split(":") if(split_artifact.size < 4) raise ArtifactMalformedException end group_id, artifact_id, version, extension = split_artifact version.upcase! if version.casecmp("latest") return group_id, artifact_id, version, extension end def create_hosted_repository_json(name) params = {:provider => "maven2"} params[:providerRole] = "org.sonatype.nexus.proxy.repository.Repository" params[:exposed] = true params[:browseable] = true params[:indexable] = true params[:repoType] = "hosted" params[:repoPolicy] = "RELEASE" params[:name] = name params[:id] = sanitize_for_id(name) params[:format] = "maven2" JSON.dump(:data => params) end def create_proxy_repository_json(name, url) params = {:provider => "maven2"} params[:providerRole] = "org.sonatype.nexus.proxy.repository.Repository" params[:exposed] = true params[:browseable] = true params[:indexable] = true params[:repoType] = "proxy" params[:repoPolicy] = "RELEASE" params[:checksumPolicy] = "WARN" params[:writePolicy] = "READ_ONLY" params[:downloadRemoteIndexes] = true params[:autoBlockActive] = true params[:name] = name params[:id] = sanitize_for_id(name) params[:remoteStorage] = {:remoteStorageUrl => url.nil? ? "http://change-me.com/" : url} JSON.dump(:data => params) end def create_user_json(params) JSON.dump(:data => params) end def create_change_password_json(params) JSON.dump(:data => params) end def create_logger_level_json(level) params = {:rootLoggerLevel => level.upcase} JSON.dump(:data => params) end def create_group_repository_json(name) params = {:id => sanitize_for_id(name)} params[:name] = name params[:provider] = "maven2" params[:exposed] = true JSON.dump(:data => params) end def create_add_to_group_repository_json(group_id, repository_to_add_id) group_repository_json = JSON.parse(get_group_repository(group_id)) repositories = group_repository_json["data"]["repositories"] repositories << {:id => sanitize_for_id(repository_to_add_id)} params = {:repositories => repositories} params[:id] = group_repository_json["data"]["id"] params[:name] = group_repository_json["data"]["name"] JSON.dump(:data => params) end def create_remove_from_group_repository_json(group_id, repository_to_remove_id) group_repository_json = JSON.parse(get_group_repository(group_id)) repositories = group_repository_json["data"]["repositories"] repositories.delete(repository_in_group?(group_id, repository_to_remove_id)) params = {:repositories => repositories} params[:id] = group_repository_json["data"]["id"] params[:name] = group_repository_json["data"]["name"] JSON.dump(:data => params) end end end