aws.rb 12 KB

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