faaso/src/faaso.cr

204 lines
5.8 KiB
Crystal
Raw Normal View History

require "./funko.cr"
require "commander"
2024-06-28 20:42:10 +00:00
require "docr"
require "docr/utils.cr"
2024-06-28 19:24:52 +00:00
require "file_utils"
require "uuid"
2024-06-29 15:29:53 +00:00
# Functions as a Service, Ops!
2024-06-28 15:41:21 +00:00
module Faaso
VERSION = "0.1.0"
2024-06-30 00:01:28 +00:00
# Ensure the faaso-net network exists
2024-06-29 21:58:49 +00:00
def self.setup_network
docker_api = Docr::API.new(Docr::Client.new)
docker_api.networks.create(Docr::Types::NetworkConfig.new(
name: "faaso-net",
check_duplicate: false,
driver: "bridge"
))
2024-06-30 00:01:28 +00:00
rescue ex : Docr::Errors::DockerAPIError
raise ex if ex.status_code != 409 # Network already exists
2024-06-29 21:58:49 +00:00
end
module Commands
class Build
@arguments : Array(String) = [] of String
@options : Commander::Options
def initialize(options, arguments)
@options = options
@arguments = arguments
end
def run
funkos = Funko.from_paths(@arguments)
funkos.each do |funko|
# Create temporary build location
tmp_dir = Path.new("tmp", UUID.random.to_s)
Dir.mkdir_p(tmp_dir) unless File.exists? tmp_dir
# Copy runtime if requested
if !funko.runtime.nil?
runtime_dir = Path.new("runtimes", funko.runtime.to_s)
if !File.exists? runtime_dir
puts "Error: runtime #{funko.runtime} not found"
next
end
Dir.glob("#{runtime_dir}/*").each { |src|
FileUtils.cp_r(src, tmp_dir)
}
2024-06-28 19:24:52 +00:00
end
# Copy funko
if funko.path.empty?
puts "Internal error: empty funko path for #{funko.name}"
next
end
Dir.glob("#{funko.path}/*").each { |src|
FileUtils.cp_r(src, tmp_dir)
}
puts "Building function... #{funko.name} in #{tmp_dir}"
2024-06-28 20:42:10 +00:00
docker_api = Docr::API.new(Docr::Client.new)
docker_api.images.build(
context: tmp_dir.to_s,
tags: ["#{funko.name}:latest"]) { }
end
end
end
class Up
@arguments : Array(String) = [] of String
@options : Commander::Options
def initialize(options, arguments)
@options = options
@arguments = arguments
end
def run
2024-06-29 15:29:53 +00:00
funkos = Funko.from_paths(@arguments)
funkos.each do |funko|
container_name = "faaso-#{funko.name}"
2024-06-28 23:07:27 +00:00
docker_api = Docr::API.new(Docr::Client.new)
images = funko.image_history
if images.empty?
puts "Error: no images available for #{funko.name}:latest"
next
end
2024-06-29 15:29:53 +00:00
# sort list of funko containers newer image first
containers = funko.containers.sort { |a, b|
(images.index(b.@image_id) || 9999) <=> (images.index(a.@image_id) || 9999)
}
# If it's already up, do nothing
# FIXME: bring back out-of-date warning
if funko.running?
puts "#{funko.name} is already up"
2024-06-29 15:29:53 +00:00
next
2024-06-28 23:07:27 +00:00
end
2024-06-28 23:57:04 +00:00
# If it is paused, unpause it
2024-06-29 15:29:53 +00:00
paused = containers.select { |container|
container.@state == "paused"
2024-06-28 23:57:04 +00:00
}
2024-06-28 23:57:58 +00:00
if paused.size > 0
2024-06-28 23:57:04 +00:00
puts "Resuming existing paused container"
docker_api.containers.unpause(paused[0].@id)
2024-06-29 15:29:53 +00:00
next
2024-06-28 23:57:04 +00:00
end
2024-06-28 23:07:27 +00:00
# If it is exited, start it
2024-06-29 15:29:53 +00:00
existing = containers.select { |container|
container.@state == "exited"
2024-06-28 23:07:27 +00:00
}
2024-06-29 15:29:53 +00:00
puts "Starting function #{funko.name}"
2024-06-28 23:57:58 +00:00
if existing.size > 0
2024-06-28 23:07:27 +00:00
puts "Restarting existing exited container"
docker_api.containers.start(existing[0].@id)
2024-06-29 15:29:53 +00:00
next
2024-06-28 23:07:27 +00:00
end
2024-06-28 23:57:04 +00:00
# Deploy from scratch
2024-06-30 00:01:28 +00:00
Faaso.setup_network # We need it
2024-06-28 23:57:04 +00:00
puts "Creating new container"
conf = Docr::Types::CreateContainerConfig.new(
image: "#{funko.name}:latest",
2024-06-29 15:29:53 +00:00
hostname: funko.name,
# Port in the container side
2024-06-29 15:29:53 +00:00
exposed_ports: {"#{funko.port}/tcp" => {} of String => String},
host_config: Docr::Types::HostConfig.new(
2024-06-29 21:58:49 +00:00
network_mode: "faaso-net",
2024-06-29 15:29:53 +00:00
port_bindings: {"#{funko.port}/tcp" => [Docr::Types::PortBinding.new(
host_port: "", # Host port, empty means random
host_ip: "127.0.0.1", # Host IP
)]}
)
2024-06-28 23:57:04 +00:00
)
2024-06-29 15:29:53 +00:00
response = docker_api.containers.create(name: container_name, config: conf)
2024-06-29 15:35:03 +00:00
response.@warnings.each { |msg| puts "Warning: #{msg}" }
2024-06-28 23:57:04 +00:00
docker_api.containers.start(response.@id)
containers = docker_api.containers.list(
all: true,
filters: {"name" => [container_name]}
)
2024-06-30 00:01:28 +00:00
(1..5).each { |_|
break if containers[0].state == "running"
sleep 0.1.seconds
}
if containers[0].state != "running"
puts "Container for #{funko.name} is not running yet"
next
end
puts "Container for #{funko.name} is running"
end
end
end
class Down
@arguments : Array(String) = [] of String
@options : Commander::Options
def initialize(options, arguments)
@options = options
@arguments = arguments
end
def run
@arguments.each do |arg|
puts "Stopping function... #{arg}"
# TODO: check if function is running
# TODO: stop function container
# TODO: delete function container
# TODO: remove route from reverse proxy
end
end
end
class Deploy
@arguments : Array(String) = [] of String
@options : Commander::Options
def initialize(options, arguments)
@options = options
@arguments = arguments
end
def run
@arguments.each do |arg|
puts "Stopping function... #{arg}"
# TODO: Everything
end
end
end
end
2024-06-28 15:41:21 +00:00
end