inspec/lib/resources/apt.rb

150 lines
4.1 KiB
Ruby
Raw Normal View History

2015-10-10 17:53:39 +00:00
# encoding: utf-8
# author: Christoph Hartmann
# author: Dominik Richter
# Verifies apt and ppa repositories
#
# Usage:
2015-10-21 17:30:01 +00:00
# describe apt('ubuntu-wine/ppa') do
2015-10-10 17:53:39 +00:00
# it { should exist }
# it { should be_enabled }
# end
#
# it also understands a ppa url
2015-10-21 17:30:01 +00:00
# describe apt('ppa:ubuntu-wine/ppa') do
2015-10-10 17:53:39 +00:00
# it { should exist }
# it { should be_enabled }
# end
#
# The following ppa formats are supported:
# - ubuntu-wine/ppa
# - ppa:ubuntu-wine/ppa
# - http://ppa.launchpad.net/juju/stable/ubuntu
#
# Install a ppa as following:
# apt-get install python-software-properties
# apt-get install software-properties-common
# add-apt-repository ppa:ubuntu-wine/ppa
2015-10-17 17:33:52 +00:00
require 'uri'
module Inspec::Resources
class AptRepository < Inspec.resource(1)
name 'apt'
desc 'Use the apt InSpec audit resource to verify Apt repositories on the Debian and Ubuntu platforms, and also PPA repositories on the Ubuntu platform.'
example "
describe apt('nginx/stable') do
it { should exist }
it { should be_enabled }
end
"
def initialize(ppa_name)
@deb_url = nil
# check if the os is ubuntu or debian
if inspec.os.debian?
@deb_url = determine_ppa_url(ppa_name)
else
# this resource is only supported on ubuntu and debian
skip_resource 'The `apt` resource is not supported on your OS yet.'
end
2015-11-27 13:02:38 +00:00
end
2015-10-10 17:53:39 +00:00
def exists?
find_repo.count > 0
end
2015-10-10 17:53:39 +00:00
def enabled?
return false if find_repo.count == 0
actives = find_repo.map { |repo| repo[:active] }
actives = actives.uniq
actives.size == 1 && actives[0] = true
end
2015-10-10 17:53:39 +00:00
def to_s
"Apt Repository #{@deb_url}"
end
2015-10-10 17:53:39 +00:00
private
2015-10-10 17:53:39 +00:00
def find_repo
read_debs.select { |repo| repo[:url] == @deb_url && repo[:type] == 'deb' }
end
2015-10-10 17:53:39 +00:00
HTTP_URL_RE = /\A#{URI::DEFAULT_PARSER.make_regexp(%w{http https})}\z/
2015-10-16 21:47:32 +00:00
# read
def read_debs
return @repo_cache if defined?(@repo_cache)
2015-10-10 17:53:39 +00:00
# load all lists
cmd = inspec.command("find /etc/apt/ -name \*.list -exec sh -c 'cat {} || echo -n' \\;")
2015-10-10 17:53:39 +00:00
# @see https://help.ubuntu.com/community/Repositories/CommandLine#Explanation_of_the_Repository_Format
@repo_cache = cmd.stdout.chomp.split("\n").each_with_object([]) do |raw_line, lines|
active = true
2015-10-10 17:53:39 +00:00
# detect if the repo is commented out
line = raw_line.gsub(/^(#\s*)*/, '')
active = false if raw_line != line
2015-10-10 17:53:39 +00:00
# eg.: deb http://archive.ubuntu.com/ubuntu/ wily main restricted
parse_repo = /^\s*(\S+)\s+"?([^ "\t\r\n\f]+)"?\s+(\S+)\s+(.*)$/.match(line)
2015-10-10 17:53:39 +00:00
# check if we got any result and the second param is an url
next if parse_repo.nil? || !parse_repo[2] =~ HTTP_URL_RE
2015-10-10 17:53:39 +00:00
# map data
repo = {
type: parse_repo[1],
url: parse_repo[2],
distro: parse_repo[3],
components: parse_repo[4].chomp.split(' '),
active: active,
}
next unless ['deb', 'deb-src'].include? repo[:type]
2015-10-10 17:53:39 +00:00
lines.push(repo)
end
2015-10-10 17:53:39 +00:00
end
# resolves ppa urls
# @see http://bazaar.launchpad.net/~ubuntu-core-dev/software-properties/main/view/head:/softwareproperties/ppa.py
def determine_ppa_url(ppa_url)
# verify if we have the url already, then just return
return ppa_url if ppa_url =~ HTTP_URL_RE
# otherwise start generating the ppa url
2015-10-10 17:53:39 +00:00
# special care if the name stats with :
ppa_url = ppa_url.split(':')[1] if ppa_url.start_with?('ppa:')
2015-10-10 17:53:39 +00:00
# parse ppa owner and repo
ppa_owner, ppa_repo = ppa_url.split('/')
ppa_repo = 'ppa' if ppa_repo.nil?
2015-10-10 17:53:39 +00:00
# construct new ppa url and return it
format('http://ppa.launchpad.net/%s/%s/ubuntu', ppa_owner, ppa_repo)
end
2015-10-10 17:53:39 +00:00
end
# for compatability with serverspec
# this is deprecated syntax and will be removed in future versions
class PpaRepository < AptRepository
name 'ppa'
2015-10-10 17:53:39 +00:00
def exists?
deprecated
super()
end
2015-10-10 17:53:39 +00:00
def enabled?
deprecated
super()
end
2015-10-10 17:53:39 +00:00
def deprecated
warn '[DEPRECATION] `ppa(reponame)` is deprecated. Please use `apt(reponame)` instead.'
end
2015-10-10 17:53:39 +00:00
end
end