aws.rb 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. # frozen_string_literal: true
  2. require 'accounts'
  3. require 'regions'
  4. require 'mime-types'
  5. require 'fog-aws'
  6. require 's3'
  7. require 'date'
  8. require 'ipaddr'
  9. require 'neo4j'
  10. require 'rds'
  11. require 'lambdas'
  12. require 'neoinfra/config'
  13. require 'neoinfra/cloudwatch'
  14. RFC_1918 = [
  15. IPAddr.new('10.0.0.0/8'),
  16. IPAddr.new('172.16.0.0/12'),
  17. IPAddr.new('192.168.0.0/16')
  18. ].freeze
  19. # NeoInfra Account information
  20. module NeoInfra
  21. # Provide informations about the accounts available
  22. class Aws
  23. def initialize
  24. @cfg = NeoInfra::Config.new
  25. neo4j_url = "http://#{@cfg.neo4j[:host]}:#{@cfg.neo4j[:port]}"
  26. Neo4j::Session.open(:server_db, neo4j_url)
  27. end
  28. def regions
  29. account = @cfg.accounts.first
  30. base_conf = {
  31. provider: 'AWS',
  32. aws_access_key_id: account[:key],
  33. aws_secret_access_key: account[:secret]
  34. }
  35. conn = Fog::Compute.new(base_conf)
  36. conn.describe_regions.data[:body]['regionInfo'].collect { |x| x['regionName'] }
  37. end
  38. def region_count
  39. Region.all.length
  40. end
  41. def az_count
  42. Az.all.length
  43. end
  44. def azs(region)
  45. account = @cfg.accounts.first
  46. base_conf = {
  47. provider: 'AWS',
  48. aws_access_key_id: account[:key],
  49. aws_secret_access_key: account[:secret],
  50. region: region
  51. }
  52. begin
  53. conn = Fog::Compute.new(base_conf)
  54. conn.describe_availability_zones.data[:body]['availabilityZoneInfo'].collect { |x| x['zoneName'] }
  55. rescue Exception => e
  56. puts "Zone couldn't load region #{region}: #{e.message}"
  57. end
  58. end
  59. def load_regions
  60. regions.each do |region|
  61. next unless Region.where(region: region).empty?
  62. r = Region.new(
  63. region: region
  64. )
  65. r.save
  66. azs(region).each do |az|
  67. next unless Az.where(az: az).empty?
  68. a = Az.new(az: az)
  69. a.save
  70. AzRegion.create(from_node: a, to_node: Region.where(region: region).first)
  71. end
  72. end
  73. end
  74. def list_buckets
  75. buckets = []
  76. Bucket.all.order('n.size DESC').each do |b|
  77. buckets << {
  78. 'name' => b.name,
  79. 'size' => b.size,
  80. 'versioning' => b.versioning,
  81. 'creation' => b.creation,
  82. 'region' => b.region.region,
  83. 'owner' => b.owner.name
  84. }
  85. end
  86. buckets
  87. end
  88. def load_buckets
  89. cw = NeoInfra::Cloudwatch.new
  90. @cfg.accounts.each do |account|
  91. base_conf = {
  92. provider: 'AWS',
  93. aws_access_key_id: account[:key],
  94. aws_secret_access_key: account[:secret]
  95. }
  96. s = Fog::Storage.new(base_conf)
  97. s.directories.each do |bucket|
  98. next unless Bucket.where(name: bucket.key).empty?
  99. begin
  100. vers = bucket.versioning?.to_s
  101. crea = bucket.creation_date.to_s
  102. rescue StandardError
  103. vers = 'unknown'
  104. crea = 'unknown'
  105. end
  106. b = Bucket.new(
  107. name: bucket.key,
  108. versioning: vers,
  109. creation: crea,
  110. size: cw.get_bucket_size(account[:key], account[:secret], bucket.location, bucket.key)
  111. )
  112. b.save
  113. BucketRegion.create(from_node: b, to_node: Region.where(region: bucket.location).first)
  114. BucketAccount.create(from_node: b, to_node: AwsAccount.where(name: account[:name]).first)
  115. end
  116. end
  117. end
  118. def load_security_groups
  119. @cfg.accounts.each do |account|
  120. base_conf = {
  121. provider: 'AWS',
  122. aws_access_key_id: account[:key],
  123. aws_secret_access_key: account[:secret]
  124. }
  125. regions.each do |region|
  126. region_conf = { region: region }
  127. begin
  128. conn = Fog::Compute.new(region_conf.merge(base_conf))
  129. rescue StandardError
  130. puts "Error loading security groups for region #{region}"
  131. next
  132. end
  133. conn.security_groups.all.each do |grp|
  134. if SecurityGroup.where(sg_id: grp.group_id).empty?
  135. g = SecurityGroup.new(
  136. sg_id: grp.group_id,
  137. name: grp.name,
  138. description: grp.description
  139. )
  140. g.save
  141. begin
  142. SecurityGroupOwner.create(from_node: g, to_node: AwsAccount.where(account_id: grp.owner_id).first)
  143. unless grp.vpc_id.nil?
  144. SecurityGroupVpc.create(from_node: g, to_node: Vpc.where(vpc_id: grp.vpc_id).first)
  145. end
  146. rescue
  147. puts "Account #{account[:name]} couldn't load the following security group:"
  148. p grp
  149. end
  150. end
  151. grp.ip_permissions.each do |iprule|
  152. next unless iprule['ipProtocol'] != '-1'
  153. iprule['ipRanges'].each do |r|
  154. to_port = if iprule['toPort'] == -1
  155. 65_535
  156. else
  157. iprule['toPort']
  158. end
  159. from_port = if iprule['fromPort'] == -1
  160. 0
  161. else
  162. iprule['fromPort']
  163. end
  164. if IpRules.where(
  165. cidr_block: r['cidrIp'],
  166. direction: 'ingress',
  167. proto: iprule['ipProtocol'],
  168. to_port: to_port,
  169. from_port: from_port
  170. ).empty?
  171. rl = IpRules.new(
  172. cidr_block: r['cidrIp'],
  173. direction: 'ingress',
  174. proto: iprule['ipProtocol'],
  175. to_port: to_port,
  176. from_port: from_port,
  177. private: RFC_1918.any? { |rfc| rfc.include?(IPAddr.new(r['cidrIp'])) }
  178. )
  179. rl.save
  180. end
  181. # TODO: remove duplicate Relationships
  182. SecurityGroupsIpRules.create(
  183. from_node: SecurityGroup.where(sg_id: grp.group_id).first,
  184. to_node: IpRules.where(
  185. cidr_block: r['cidrIp'],
  186. direction: 'ingress',
  187. proto: iprule['ipProtocol'],
  188. to_port: to_port,
  189. from_port: from_port,
  190. private: RFC_1918.any? { |rfc| rfc.include?(IPAddr.new(r['cidrIp'])) }
  191. ).first
  192. )
  193. end
  194. end
  195. end
  196. end
  197. end
  198. end
  199. def list_lambdas
  200. lambdas = []
  201. Lambda.all.each do |l|
  202. lambdas << {
  203. 'name' => l.name,
  204. 'runtime' => l.runtime,
  205. 'handler' => l.handler,
  206. 'lambda_timeout' => l.lambda_timeout,
  207. 'memorysize' => l.memorysize,
  208. 'last_modified' => l.last_modified,
  209. 'region' => l.region.region,
  210. 'owner' => l.owner.name
  211. }
  212. end
  213. lambdas
  214. end
  215. def list_rds
  216. rds = []
  217. Rds.all.each do |r|
  218. rds << {
  219. 'name' => r.name,
  220. 'size' => r.size,
  221. 'engine' => r.engine,
  222. 'engine_version' => r.engine_version,
  223. 'multi_az' => r.multi_az,
  224. 'endpoint' => r.endpoint,
  225. 'port' => r.port,
  226. 'allocated_storage' => r.allocated_storage,
  227. 'owner' => r.owner.name,
  228. 'az' => r.az.az,
  229. }
  230. end
  231. rds
  232. end
  233. def load_lambda
  234. @cfg.accounts.each do |account|
  235. base_conf = {
  236. aws_access_key_id: account[:key],
  237. aws_secret_access_key: account[:secret]
  238. }
  239. regions.each do |region|
  240. region_conf = { region: region }
  241. begin
  242. lambdas = Fog::AWS::Lambda.new(region_conf.merge(base_conf))
  243. lambdas.list_functions.data[:body]['Functions'].each do |f|
  244. next unless Lambda.where(name: f['FunctionArn']).empty?
  245. l = Lambda.new(
  246. name: f['FunctionName'],
  247. runtime: f['Runtime'],
  248. lambda_timeout: f['Timeout'],
  249. handler: f['Handler'],
  250. memorysize: f['MemorySize'],
  251. arn: f['FunctionArn'],
  252. codesize: f['CodeSize'],
  253. last_modified: f['LastModified']
  254. )
  255. l.save
  256. LambdaAccount.create(from_node: l, to_node: AwsAccount.where(name: account[:name]).first)
  257. LambdaRegion.create(from_node: l, to_node: Region.where(region: region).first)
  258. end
  259. rescue Exception => e
  260. puts "Error with #{region}: #{e.message}"
  261. next
  262. end
  263. end
  264. end
  265. end
  266. def list_dynamos
  267. dynamos = []
  268. Dynamo.all.order('n.sizebytes DESC').each do |d|
  269. dynamos << {
  270. 'name' => d.name,
  271. 'size' => d.sizebytes,
  272. 'itemcount' => d.itemcount,
  273. 'status' => d.status,
  274. 'creation' => d.creation,
  275. 'region' => d.region.region,
  276. 'owner' => d.owner.name
  277. }
  278. end
  279. dynamos
  280. end
  281. def load_dynamo
  282. @cfg.accounts.each do |account|
  283. base_conf = {
  284. aws_access_key_id: account[:key],
  285. aws_secret_access_key: account[:secret]
  286. }
  287. regions.each do |region|
  288. region_conf = { region: region }
  289. begin
  290. dyns = Fog::AWS::DynamoDB.new(region_conf.merge(base_conf))
  291. dyns.list_tables.data[:body]['TableNames'].each do |table|
  292. tb = dyns.describe_table(table).data[:body]['Table']
  293. next unless Dynamo.where(name: table['TableId']).empty?
  294. d = Dynamo.new(
  295. tableid: tb['TableId'],
  296. name: tb['TableName'],
  297. creation: Time.at(tb['CreationDateTime']).to_datetime.strftime('%F %H:%M:%S %Z'),
  298. arn: tb['TableArn'],
  299. itemcount: tb['ItemCount'],
  300. sizebytes: tb['TableSizeBytes'],
  301. status: tb['TableStatus'],
  302. readcap: tb['ProvisionedThroughput']['ReadCapacityUnits'],
  303. writecap: tb['ProvisionedThroughput']['WriteCapacityUnits'],
  304. capdecreases: tb['ProvisionedThroughput']['NumberOfDecreasesToday']
  305. )
  306. d.save
  307. DynamoAccount.create(from_node: d, to_node: AwsAccount.where(name: account[:name]).first)
  308. DynamoRegion.create(from_node: d, to_node: Region.where(region: region).first)
  309. end
  310. rescue Exception => e
  311. puts "Could not list Dynamos for region: #{region}: #{e.message}"
  312. next
  313. end
  314. end
  315. # dyns.list_tables.each do |table|
  316. # p table
  317. # end
  318. end
  319. end
  320. def load_rds
  321. @cfg.accounts.each do |account|
  322. base_conf = {
  323. aws_access_key_id: account[:key],
  324. aws_secret_access_key: account[:secret]
  325. }
  326. regions.each do |region|
  327. region_conf = { region: region }
  328. s = Fog::AWS::RDS.new(region_conf.merge(base_conf))
  329. s.servers.each do |rds|
  330. next unless Rds.where(name: rds.id).empty?
  331. r = Rds.new(
  332. name: rds.id,
  333. size: rds.flavor_id,
  334. engine: rds.engine,
  335. engine_version: rds.engine_version,
  336. multi_az: rds.multi_az.to_s,
  337. endpoint: rds.endpoint['Address'],
  338. port: rds.endpoint['Port'],
  339. allocated_storage: rds.allocated_storage
  340. )
  341. r.save
  342. begin
  343. RdsAz.create(from_node: r, to_node: Az.where(az: rds.availability_zone).first)
  344. RdsAccount.create(from_node: r, to_node: AwsAccount.where(name: account[:name]).first)
  345. rescue Exception => e
  346. puts "Account #{account[:name]} couldn't load the following rds: #{e.message}"
  347. p r
  348. next
  349. end
  350. end
  351. end
  352. end
  353. end
  354. end
  355. end