From 3ae6437425c7aa6df3a04a208175f28a77c6ade2 Mon Sep 17 00:00:00 2001 From: zzak Date: Sun, 3 Sep 2023 20:34:55 +0900 Subject: [PATCH] wip: poc: buildkite-builder --- .buildkite/.keep | 0 Gemfile | 3 +- Gemfile.lock | 4 + buildkite-config-initial-pipeline.yml | 4 +- pipeline-generate | 443 +----------------------- pipelines/rails-ci/pipeline.rb | 464 ++++++++++++++++++++++++++ rails-initial-pipeline.yml | 5 +- 7 files changed, 488 insertions(+), 435 deletions(-) create mode 100644 .buildkite/.keep create mode 100644 pipelines/rails-ci/pipeline.rb diff --git a/.buildkite/.keep b/.buildkite/.keep new file mode 100644 index 00000000..e69de29b diff --git a/Gemfile b/Gemfile index c48b9a5b..caab62c4 100644 --- a/Gemfile +++ b/Gemfile @@ -1,6 +1,7 @@ source "https://rubygems.org" +gem "autotest" gem "buildkit" +gem "buildkite-builder" gem "diffy" gem "rake" -gem "autotest" diff --git a/Gemfile.lock b/Gemfile.lock index 32b5da77..ee3deac5 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -7,6 +7,8 @@ GEM minitest-autotest (~> 1.0) buildkit (1.5.0) sawyer (>= 0.6) + buildkite-builder (4.2.4) + rainbow (>= 3) diffy (3.4.2) faraday (2.7.7) faraday-net_http (>= 2.0, < 3.1) @@ -20,6 +22,7 @@ GEM minitest (~> 5.16) path_expander (1.1.1) public_suffix (5.0.1) + rainbow (3.1.1) rake (13.0.6) ruby2_keywords (0.0.5) sawyer (0.9.2) @@ -33,6 +36,7 @@ PLATFORMS DEPENDENCIES autotest buildkit + buildkite-builder diffy rake diff --git a/buildkite-config-initial-pipeline.yml b/buildkite-config-initial-pipeline.yml index e361abca..03f1b217 100644 --- a/buildkite-config-initial-pipeline.yml +++ b/buildkite-config-initial-pipeline.yml @@ -53,7 +53,7 @@ steps: message: "[${BUILDKITE_BRANCH}] ${BUILDKITE_MESSAGE}" branch: "main" env: - CONFIG_REPO: "${BUILDKITE_PULL_REQUEST_REPO}" + CONFIG_REPO: "${BUILDKITE_REPO}" CONFIG_BRANCH: "${BUILDKITE_BRANCH}" BUILDKITE_CONFIG_TRIGGER: true # This variable can be used downstream to avoid things like "if branch==main do Y" - trigger: "rails-ci" @@ -63,6 +63,6 @@ steps: message: "[${BUILDKITE_BRANCH} / 6-1-stable] ${BUILDKITE_MESSAGE}" branch: "6-1-stable" env: - CONFIG_REPO: "${BUILDKITE_PULL_REQUEST_REPO}" + CONFIG_REPO: "${BUILDKITE_REPO}" CONFIG_BRANCH: "${BUILDKITE_BRANCH}" BUILDKITE_CONFIG_TRIGGER: true # This variable can be used downstream to avoid things like "if branch==main do Y" diff --git a/pipeline-generate b/pipeline-generate index 0b1f025f..60b2f7f6 100755 --- a/pipeline-generate +++ b/pipeline-generate @@ -1,443 +1,24 @@ #!/usr/bin/env ruby # frozen_string_literal: true -require "json" -require "net/http" -require "pathname" -require "yaml" +def run(cmd, env={}, value=false) + io = IO.popen(env, cmd) + output = io.read + io.close -STANDARD_QUEUES = [nil, "default", "builder"] + raise output unless $?.success? -# If the pipeline is running in a non-standard queue, default to -# running everything in that queue. -unless STANDARD_QUEUES.include?(ENV["BUILDKITE_AGENT_META_DATA_QUEUE"]) - ENV["QUEUE"] ||= ENV["BUILDKITE_AGENT_META_DATA_QUEUE"] + return output if value end -BUILD_QUEUE = ENV["BUILD_QUEUE"] || ENV["QUEUE"] || "builder" -RUN_QUEUE = ENV["RUN_QUEUE"] || ENV["QUEUE"] || "default" +env = {} -IMAGE_BASE = ENV["DOCKER_IMAGE"] || "973266071021.dkr.ecr.us-east-1.amazonaws.com/#{"#{BUILD_QUEUE}-" unless STANDARD_QUEUES.include?(BUILD_QUEUE)}builds" - -BASE_BRANCH = ([ENV["BUILDKITE_PULL_REQUEST_BASE_BRANCH"], ENV["BUILDKITE_BRANCH"], "main"] - [""]).first -LOCAL_BRANCH = ([ENV["BUILDKITE_BRANCH"], "main"] - [""]).first -PULL_REQUEST = ([ENV["BUILDKITE_PULL_REQUEST"]] - ["false"]).first - -BUILD_ID = ENV["BUILDKITE_BUILD_ID"] -REBUILD_ID = ([ENV["BUILDKITE_REBUILT_FROM_BUILD_ID"]] - [""]).first - -MAINLINE = LOCAL_BRANCH == "main" || LOCAL_BRANCH =~ /\A[0-9-]+(?:-stable)?\z/ - -DOCKER_COMPOSE_PLUGIN = "docker-compose#v3.7.0" -ARTIFACTS_PLUGIN = "artifacts#v1.2.0" - -REPO_ROOT = Pathname.new(ARGV.shift || File.expand_path("../..", __FILE__)) - -REPO_ROOT.join("rails.gemspec").read =~ /required_ruby_version[^0-9]+([0-9]+\.[0-9]+)/ -RUBY_MINORS = %w(2.4 2.5 2.6 2.7 3.0 3.1 3.2).map { |v| Gem::Version.new(v) } -MIN_RUBY = Gem::Version.new($1 || "2.0") - -RAILS_VERSION = Gem::Version.new(File.read(REPO_ROOT.join("RAILS_VERSION"))) -BUNDLER = - case RAILS_VERSION - when Gem::Requirement.new("< 5.0") - "< 2" - when Gem::Requirement.new("< 6.1") - "< 2.2.10" - end -RUBYGEMS = - case RAILS_VERSION - when Gem::Requirement.new("< 5.0") - "2.6.13" - when Gem::Requirement.new("< 6.1") - "3.2.9" - end - -MAX_RUBY = - case RAILS_VERSION - when Gem::Requirement.new("< 5.1") - Gem::Version.new("2.4") - when Gem::Requirement.new("< 5.2") - Gem::Version.new("2.5") - when Gem::Requirement.new("< 6.0") - Gem::Version.new("2.6") - when Gem::Requirement.new("< 6.1") - Gem::Version.new("2.7") - end - - -RUBIES = [] -SOFT_FAIL = [] - -RUBY_MINORS.select { |v| v >= MIN_RUBY }.each do |v| - image = "ruby:#{v}" - - if MAX_RUBY && v > MAX_RUBY && !(MAX_RUBY.approximate_recommendation === v) - SOFT_FAIL << image - else - RUBIES << image - end -end - -ONE_RUBY = RUBIES.last || SOFT_FAIL.last - -MASTER_RUBY = "rubylang/ruby:master-nightly-jammy" -SOFT_FAIL << MASTER_RUBY - -# Adds yjit: onto the master ruby image string so we -# know when to turn on YJIT via the environment variable. -# Same as master ruby, we want this to soft fail. -YJIT_RUBY = "yjit:#{MASTER_RUBY}" -SOFT_FAIL << YJIT_RUBY - -# Run steps for newer Rubies first. -RUBIES.reverse! -SOFT_FAIL.reverse! - -# Run soft-failing Ruby steps last. -RUBIES.concat SOFT_FAIL - -STEPS = [] - -def image_name_for(ruby, suffix = BUILD_ID, short: false) - ruby = ruby_image(ruby) - - tag = "#{mangle_name(ruby)}-#{suffix}" - - if short - tag - else - "#{IMAGE_BASE}:#{tag}" - end -end - -def mangle_name(name) - name.tr("^A-Za-z0-9", "-") -end - -# YJIT uses the same image as ruby-trunk because it's turned on -# via an ENV var. This needs to remove the `yjit:` added onto the -# front because otherwise it's not a valid image. -def ruby_image(ruby) - if ruby == YJIT_RUBY - ruby.sub("yjit:", "") - else - ruby - end -end - -# A shortened version of the name for the Buildkite label. -def short_ruby(ruby) - if ruby == MASTER_RUBY - "master" - elsif ruby == YJIT_RUBY - "yjit" - else - ruby.sub(/^ruby:|:latest$/, "") - end -end - -def step_for(subdirectory, rake_task, ruby: nil, service: "default", pre_steps: []) - return unless REPO_ROOT.join(subdirectory).exist? - - label = +"#{subdirectory} #{rake_task.sub(/[:_]test|test:/, "")}" - label.sub!(/ test/, "") - if ruby - label << " (#{short_ruby(ruby)})" - end - - if rake_task.start_with?("mysql2:") || (RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") && rake_task.start_with?("trilogy:")) - rake_task = "db:mysql:rebuild #{rake_task}" - elsif rake_task.start_with?("postgresql:") - rake_task = "db:postgresql:rebuild #{rake_task}" - end - - env = { - "IMAGE_NAME" => image_name_for(ruby || ONE_RUBY), - } - - # If we have YJIT_RUBY set the environment variable - # to turn it on. - if ruby == YJIT_RUBY - env["RUBY_YJIT_ENABLE"] = "1" - end - - if !pre_steps.empty? - env["PRE_STEPS"] = pre_steps.join(" && ") - end - command = "rake #{rake_task}" - - timeout = 30 - - group = - if rake_task.include?("isolated") - "isolated" - else - ruby || ONE_RUBY - end - - if RAILS_VERSION < Gem::Version.new("5.x") - env["MYSQL_IMAGE"] = "mysql:5.6" - elsif RAILS_VERSION < Gem::Version.new("6.x") - env["MYSQL_IMAGE"] = "mysql:5.7" - end - - if RAILS_VERSION < Gem::Version.new("5.2.x") - env["POSTGRES_IMAGE"] = "postgres:9.6-alpine" - end - - hash = { - "label" => label, - "depends_on" => "docker-image-#{ruby_image(ruby || ONE_RUBY).gsub(/\W/, "-")}", - "command" => command, - "group" => group, - "plugins" => [ - { - ARTIFACTS_PLUGIN => { - "download" => [".buildkite/*", ".buildkite/**/*"], - }, - }, - { - DOCKER_COMPOSE_PLUGIN => { - "env" => [ - "PRE_STEPS", - "RACK" - ], - "run" => service, - "pull" => service, - "config" => ".buildkite/docker-compose.yml", - "shell" => ["runner", subdirectory], - }, - }, - ], - "env" => env, - "timeout_in_minutes" => timeout, - "soft_fail" => SOFT_FAIL.include?(ruby), - "agents" => { "queue" => RUN_QUEUE }, - "artifact_paths" => ["test-reports/*/*.xml"], - "retry" => { "automatic" => { "exit_status" => -1, "limit" => 2 } }, - } - - yield hash if block_given? - - STEPS << hash -end - -def steps_for(subdirectory, rake_task, service: "default", pre_steps: [], &block) - RUBIES.each do |ruby| - step_for(subdirectory, rake_task, ruby: ruby, service: service, pre_steps: pre_steps, &block) - end -end - -# GROUP 1: Runs additional isolated tests for non-PR builds -%w( - actionpack test default - actionmailer test default - activemodel test default - activesupport test default - actionview test default - actiontext test default - activejob test default - activerecord mysql2:test mysqldb - activerecord trilogy:test mysqldb - activerecord postgresql:test postgresdb - activerecord sqlite3:test default -).each_slice(3) do |dir, task, service| - next if RAILS_VERSION < Gem::Version.new("7.1.0.alpha") && task == "trilogy:test" - - steps_for(dir, task, service: service) - - next unless MAINLINE - - if dir == "activerecord" - step_for(dir, task.sub(":test", ":isolated_test"), service: service) do |x| - x["parallelism"] = 5 if REPO_ROOT.join("activerecord/Rakefile").read.include?("BUILDKITE_PARALLEL") - end - elsif dir == "actiontext" - # added during 7.1 development on main - if REPO_ROOT.join("actiontext/Rakefile").read.include?("task :isolated") - step_for(dir, "#{task}:isolated", service: service) - end - else - step_for(dir, "#{task}:isolated", service: service) - end -end - -# GROUP 2: No isolated tests, runs for each supported ruby -%w( - activestorage test default - actionmailbox test default - guides test default -).each_slice(3) do |dir, task, service| - steps_for(dir, task, service: service) -end - -# GROUP 3: Special cases - -steps_for("actioncable", "test", service: "postgresdb", pre_steps: ["cd ./activerecord", "bundle exec rake db:postgresql:rebuild", "cd -"]) - -if RAILS_VERSION >= Gem::Version.new("5.1.x") - step_for("activerecord", "sqlite3_mem:test", service: "default") -end -if RAILS_VERSION >= Gem::Version.new("6.1.x") - step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| - x["label"] += " [prepared_statements]" - x["env"]["MYSQL_PREPARED_STATEMENTS"] = "true" - end -end -step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| - x["label"] += " [mysql_5_7]" - x["env"]["MYSQL_IMAGE"] = "mysql:5.7" -end -if RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") - step_for("activerecord", "trilogy:test", service: "mysqldb") do |x| - x["label"] += " [mysql_5_7]" - x["env"]["MYSQL_IMAGE"] = "mysql:5.7" - end -end -if RAILS_VERSION >= Gem::Version.new("5.x") - step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| - x["label"] += " [mariadb]" - x["env"]["MYSQL_IMAGE"] = - if RAILS_VERSION < Gem::Version.new("6.x") - "mariadb:10.2" - else - "mariadb:latest" - end - end -end -if RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") - step_for("activerecord", "trilogy:test", service: "mysqldb") do |x| - x["label"] += " [mariadb]" - x["env"]["MYSQL_IMAGE"] = "mariadb:latest" - end -end -steps_for("actioncable", "test:integration", service: "default") do |x| - if RAILS_VERSION < Gem::Version.new("6.x") - x["soft_fail"] = true - else - x["retry"] = { "automatic" => { "limit" => 3 } } - end -end -if REPO_ROOT.join("actionview/Rakefile").read.include?("task :ujs") - step_for("actionview", "test:ujs", service: "actionview") do |x| - x["retry"] = { "automatic" => { "limit" => 3 } } - end -end -steps_for("activejob", "test:integration", service: "activejob") do |x| - # Enable soft_fail until the problem in queue_classic is solved. - # https://github.com/rails/rails/pull/37517#issuecomment-545370408 - x["soft_fail"] = true # if RAILS_VERSION < Gem::Version.new("5.x") -end -steps_for("railties", "test", service: "railties") do |x| - x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") -end - -step_for("actionpack", "test", service: "default", pre_steps: ["bundle install"]) do |x| - x["label"] += " [rack-2]" - x["env"]["RACK"] = "~> 2.0" -end - -step_for("railties", "test", service: "railties", pre_steps: ["bundle install"]) do |x| - x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") - x["label"] += " [rack-2]" - x["env"]["RACK"] = "~> 2.0" -end - -step_for("actionpack", "test", service: "default", pre_steps: ["rm Gemfile.lock", "bundle install"]) do |x| - x["label"] += " [rack-head]" - x["env"]["RACK"] = "head" - x["soft_fail"] = true +if ENV["BUILDKITE_PIPELINE_NAME"] == "rails-ci" || ENV["BUILDKITE_PIPELINE_NAME"] == "zzak/rails" + env["BUNDLE_GEMFILE"] = ".buildkite/Gemfile" end -step_for("railties", "test", service: "railties", pre_steps: ["rm Gemfile.lock", "bundle install"]) do |x| - x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") - x["label"] += " [rack-head]" - x["env"]["RACK"] = "head" - x["soft_fail"] = true -end - -# Ugly hacks to just get the build passing for now -STEPS.find { |s| s["label"] == "activestorage (2.2)" }&.tap do |s| - s["soft_fail"] = true -end - -# Bug report templates -STEPS.select { |s| s["label"] =~ /^guides/ }.each do |s| - s["soft_fail"] = true -end -if RAILS_VERSION < Gem::Version.new("7.x") && RAILS_VERSION >= Gem::Version.new("6.1") - STEPS.delete_if { |s| s["label"] == "guides (2.7)" || s["label"] == "guides (3.0)" } -end -STEPS.delete_if { |s| s["label"] =~ /^guides/ } if RAILS_VERSION < Gem::Version.new("7.0") - -### +run "bundle install", env -STEPS.sort_by! do |step| - [ - -step["timeout_in_minutes"], - step["group"] == "isolated" ? 2 : 1, - step["command"].include?("test:") ? 2 : 1, - step["label"], - ] -end - -groups = STEPS.group_by { |s| s.delete("group") }.map do |group, steps| - { "group" => group, "steps" => steps } -end +pipeline = run "bundle exec buildkite-builder preview rails-ci", env, true -puts YAML.dump("steps" => [ - { - "group" => "build", - "steps" => [ - *(RUBIES - [YJIT_RUBY]).map do |ruby| - { - "label" => ":docker: #{ruby}", - "key" => "docker-image-#{ruby.gsub(/\W/, "-")}", - "plugins" => [ - { - ARTIFACTS_PLUGIN => { - "download" => [".dockerignore", ".buildkite/*", ".buildkite/**/*"], - }, - }, - { - DOCKER_COMPOSE_PLUGIN => { - "build" => "base", - "config" => ".buildkite/docker-compose.yml", - "env" => [ - "PRE_STEPS", - "RACK" - ], - "image-name" => image_name_for(ruby, short: true), - "cache-from" => [ - REBUILD_ID && "base:" + image_name_for(ruby, REBUILD_ID), - PULL_REQUEST && "base:" + image_name_for(ruby, "pr-#{PULL_REQUEST}"), - LOCAL_BRANCH && LOCAL_BRANCH !~ /:/ && "base:" + image_name_for(ruby, "br-#{LOCAL_BRANCH}"), - BASE_BRANCH && "base:" + image_name_for(ruby, "br-#{BASE_BRANCH}"), - "base:" + image_name_for(ruby, "br-main"), - ].grep(String).uniq, - "push" => [ - LOCAL_BRANCH =~ /:/ ? - "base:" + image_name_for(ruby, "pr-#{PULL_REQUEST}") : - "base:" + image_name_for(ruby, "br-#{LOCAL_BRANCH}"), - ], - "image-repository" => IMAGE_BASE, - }, - }, - ], - "env" => { - "BUNDLER" => BUNDLER, - "RUBYGEMS" => RUBYGEMS, - "RUBY_IMAGE" => ruby_image(ruby), - "encrypted_0fb9444d0374_key" => nil, - "encrypted_0fb9444d0374_iv" => nil, - }, - "timeout_in_minutes" => 15, - "soft_fail" => SOFT_FAIL.include?(ruby), - "agents" => { "queue" => BUILD_QUEUE }, - } - end, - ], - }, - *groups, -]) +puts pipeline diff --git a/pipelines/rails-ci/pipeline.rb b/pipelines/rails-ci/pipeline.rb new file mode 100644 index 00000000..fc11489d --- /dev/null +++ b/pipelines/rails-ci/pipeline.rb @@ -0,0 +1,464 @@ + +require "json" +require "net/http" +require "pathname" +require "yaml" + + +STANDARD_QUEUES = [nil, "default", "builder"] + +# If the pipeline is running in a non-standard queue, default to +# running everything in that queue. +unless STANDARD_QUEUES.include?(ENV["BUILDKITE_AGENT_META_DATA_QUEUE"]) + ENV["QUEUE"] ||= ENV["BUILDKITE_AGENT_META_DATA_QUEUE"] +end + +BUILD_QUEUE = ENV["BUILD_QUEUE"] || ENV["QUEUE"] || "builder" +RUN_QUEUE = ENV["RUN_QUEUE"] || ENV["QUEUE"] || "default" + +IMAGE_BASE = ENV["DOCKER_IMAGE"] || "973266071021.dkr.ecr.us-east-1.amazonaws.com/#{"#{BUILD_QUEUE}-" unless STANDARD_QUEUES.include?(BUILD_QUEUE)}builds" + +BASE_BRANCH = ([ENV["BUILDKITE_PULL_REQUEST_BASE_BRANCH"], ENV["BUILDKITE_BRANCH"], "main"] - [""]).first +LOCAL_BRANCH = ([ENV["BUILDKITE_BRANCH"], "main"] - [""]).first +PULL_REQUEST = ([ENV["BUILDKITE_PULL_REQUEST"]] - ["false"]).first + +BUILD_ID = ENV["BUILDKITE_BUILD_ID"] +REBUILD_ID = ([ENV["BUILDKITE_REBUILT_FROM_BUILD_ID"]] - [""]).first + +MAINLINE = LOCAL_BRANCH == "main" || LOCAL_BRANCH =~ /\A[0-9-]+(?:-stable)?\z/ + +DOCKER_COMPOSE_PLUGIN = "docker-compose#v3.7.0" +ARTIFACTS_PLUGIN = "artifacts#v1.2.0" + +#REPO_ROOT = Pathname.new(ARGV.shift || File.expand_path("../..", __FILE__)) +if ENV["BUILDKITE_PIPELINE_NAME"] == "rails-ci" || ENV["BUILDKITE_PIPELINE_NAME"] == "zzak/rails" + REPO_ROOT = Pathname.new(Dir.pwd) +else + REPO_ROOT = Pathname.new(Dir.pwd) + "tmp/rails" +end + +REPO_ROOT.join("rails.gemspec").read =~ /required_ruby_version[^0-9]+([0-9]+\.[0-9]+)/ +RUBY_MINORS = %w(2.4 2.5 2.6 2.7 3.0 3.1 3.2).map { |v| Gem::Version.new(v) } +MIN_RUBY = Gem::Version.new($1 || "2.0") + +RAILS_VERSION = Gem::Version.new(File.read(REPO_ROOT.join("RAILS_VERSION"))) +BUNDLER = + case RAILS_VERSION + when Gem::Requirement.new("< 5.0") + "< 2" + when Gem::Requirement.new("< 6.1") + "< 2.2.10" + end +RUBYGEMS = + case RAILS_VERSION + when Gem::Requirement.new("< 5.0") + "2.6.13" + when Gem::Requirement.new("< 6.1") + "3.2.9" + end + +MAX_RUBY = + case RAILS_VERSION + when Gem::Requirement.new("< 5.1") + Gem::Version.new("2.4") + when Gem::Requirement.new("< 5.2") + Gem::Version.new("2.5") + when Gem::Requirement.new("< 6.0") + Gem::Version.new("2.6") + when Gem::Requirement.new("< 6.1") + Gem::Version.new("2.7") + end + + +RUBIES = [] +SOFT_FAIL = [] + +RUBY_MINORS.select { |v| v >= MIN_RUBY }.each do |v| + image = "ruby:#{v}" + + if MAX_RUBY && v > MAX_RUBY && !(MAX_RUBY.approximate_recommendation === v) + SOFT_FAIL << image + else + RUBIES << image + end +end + +ONE_RUBY = RUBIES.last || SOFT_FAIL.last + +MASTER_RUBY = "rubylang/ruby:master-nightly-jammy" +SOFT_FAIL << MASTER_RUBY + +# Adds yjit: onto the master ruby image string so we +# know when to turn on YJIT via the environment variable. +# Same as master ruby, we want this to soft fail. +YJIT_RUBY = "yjit:#{MASTER_RUBY}" +SOFT_FAIL << YJIT_RUBY + +# Run steps for newer Rubies first. +RUBIES.reverse! +SOFT_FAIL.reverse! + +# Run soft-failing Ruby steps last. +RUBIES.concat SOFT_FAIL + +STEPS = [] + +def image_name_for(ruby, suffix = BUILD_ID, short: false) + ruby = ruby_image(ruby) + + tag = "#{mangle_name(ruby)}-#{suffix}" + + if short + tag + else + "#{IMAGE_BASE}:#{tag}" + end +end + +def mangle_name(name) + name.tr("^A-Za-z0-9", "-") +end + +# YJIT uses the same image as ruby-trunk because it's turned on +# via an ENV var. This needs to remove the `yjit:` added onto the +# front because otherwise it's not a valid image. +def ruby_image(ruby) + if ruby == YJIT_RUBY + ruby.sub("yjit:", "") + else + ruby + end +end + +# A shortened version of the name for the Buildkite label. +def short_ruby(ruby) + if ruby == MASTER_RUBY + "master" + elsif ruby == YJIT_RUBY + "yjit" + else + ruby.sub(/^ruby:|:latest$/, "") + end +end + +def step_for(subdirectory, rake_task, ruby: nil, service: "default", pre_steps: []) + return unless REPO_ROOT.join(subdirectory).exist? + + label = +"#{subdirectory} #{rake_task.sub(/[:_]test|test:/, "")}" + label.sub!(/ test/, "") + if ruby + label << " (#{short_ruby(ruby)})" + end + + if rake_task.start_with?("mysql2:") || (RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") && rake_task.start_with?("trilogy:")) + rake_task = "db:mysql:rebuild #{rake_task}" + elsif rake_task.start_with?("postgresql:") + rake_task = "db:postgresql:rebuild #{rake_task}" + end + + env = { + "IMAGE_NAME" => image_name_for(ruby || ONE_RUBY), + } + + # If we have YJIT_RUBY set the environment variable + # to turn it on. + if ruby == YJIT_RUBY + env["RUBY_YJIT_ENABLE"] = "1" + end + + if !pre_steps.empty? + env["PRE_STEPS"] = pre_steps.join(" && ") + end + command = "rake #{rake_task}" + + timeout = 30 + + group = + if rake_task.include?("isolated") + "isolated" + else + ruby || ONE_RUBY + end + + if RAILS_VERSION < Gem::Version.new("5.x") + env["MYSQL_IMAGE"] = "mysql:5.6" + elsif RAILS_VERSION < Gem::Version.new("6.x") + env["MYSQL_IMAGE"] = "mysql:5.7" + end + + if RAILS_VERSION < Gem::Version.new("5.2.x") + env["POSTGRES_IMAGE"] = "postgres:9.6-alpine" + end + + hash = { + "label" => label, + "depends_on" => "docker-image-#{ruby_image(ruby || ONE_RUBY).gsub(/\W/, "-")}", + "command" => command, + "group" => group, + "plugins" => [ + { + ARTIFACTS_PLUGIN => { + "download" => [".buildkite/*", ".buildkite/**/*"], + }, + }, + { + DOCKER_COMPOSE_PLUGIN => { + "env" => [ + "PRE_STEPS", + "RACK" + ], + "run" => service, + "pull" => service, + "config" => ".buildkite/docker-compose.yml", + "shell" => ["runner", subdirectory], + }, + }, + ], + "env" => env, + "timeout_in_minutes" => timeout, + "soft_fail" => SOFT_FAIL.include?(ruby), + "agents" => { "queue" => RUN_QUEUE }, + "artifact_paths" => ["test-reports/*/*.xml"], + "retry" => { "automatic" => { "exit_status" => -1, "limit" => 2 } }, + } + + yield hash if block_given? + + STEPS << hash +end + +def steps_for(subdirectory, rake_task, service: "default", pre_steps: [], &block) + RUBIES.each do |ruby| + step_for(subdirectory, rake_task, ruby: ruby, service: service, pre_steps: pre_steps, &block) + end +end + +# GROUP 1: Runs additional isolated tests for non-PR builds +%w( + actionpack test default + actionmailer test default + activemodel test default + activesupport test default + actionview test default + activejob test default + activerecord mysql2:test mysqldb + activerecord trilogy:test mysqldb + activerecord postgresql:test postgresdb + activerecord sqlite3:test default +).each_slice(3) do |dir, task, service| + next if RAILS_VERSION < Gem::Version.new("7.1.0.alpha") && task == "trilogy:test" + + steps_for(dir, task, service: service) + + next unless MAINLINE + + if dir == "activerecord" + step_for(dir, task.sub(":test", ":isolated_test"), service: service) do |x| + x["parallelism"] = 5 if REPO_ROOT.join("activerecord/Rakefile").read.include?("BUILDKITE_PARALLEL") + end + elsif dir == "actiontext" + # added during 7.1 development on main + if REPO_ROOT.join("actiontext/Rakefile").read.include?("task :isolated") + step_for(dir, "#{task}:isolated", service: service) + end + else + step_for(dir, "#{task}:isolated", service: service) + end +end + +# GROUP 2: No isolated tests, runs for each supported ruby +%w( + actioncable test postgresdb + activestorage test default + actionmailbox test default + guides test default +).each_slice(3) do |dir, task, service| + steps_for(dir, task, service: service) +end + +# GROUP 3: Special cases + +if RAILS_VERSION >= Gem::Version.new("5.1.x") + step_for("activerecord", "sqlite3_mem:test", service: "default") +end +if RAILS_VERSION >= Gem::Version.new("6.1.x") + step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| + x["label"] += " [prepared_statements]" + x["env"]["MYSQL_PREPARED_STATEMENTS"] = "true" + end +end +step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| + x["label"] += " [mysql_5_7]" + x["env"]["MYSQL_IMAGE"] = "mysql:5.7" +end +if RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") + step_for("activerecord", "trilogy:test", service: "mysqldb") do |x| + x["label"] += " [mysql_5_7]" + x["env"]["MYSQL_IMAGE"] = "mysql:5.7" + end +end +if RAILS_VERSION >= Gem::Version.new("5.x") + step_for("activerecord", "mysql2:test", service: "mysqldb") do |x| + x["label"] += " [mariadb]" + x["env"]["MYSQL_IMAGE"] = + if RAILS_VERSION < Gem::Version.new("6.x") + "mariadb:10.2" + else + "mariadb:latest" + end + end +end +if RAILS_VERSION >= Gem::Version.new("7.1.0.alpha") + step_for("activerecord", "trilogy:test", service: "mysqldb") do |x| + x["label"] += " [mariadb]" + x["env"]["MYSQL_IMAGE"] = "mariadb:latest" + end +end +steps_for("actioncable", "test:integration", service: "default") do |x| + if RAILS_VERSION < Gem::Version.new("6.x") + x["soft_fail"] = true + else + x["retry"] = { "automatic" => { "limit" => 3 } } + end +end +if REPO_ROOT.join("actionview/Rakefile").read.include?("task :ujs") + step_for("actionview", "test:ujs", service: "actionview") do |x| + x["retry"] = { "automatic" => { "limit" => 3 } } + end +end +steps_for("activejob", "test:integration", service: "activejob") do |x| + # Enable soft_fail until the problem in queue_classic is solved. + # https://github.com/rails/rails/pull/37517#issuecomment-545370408 + x["soft_fail"] = true # if RAILS_VERSION < Gem::Version.new("5.x") +end +steps_for("railties", "test", service: "railties") do |x| + x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") +end + +step_for("actionpack", "test", service: "default", pre_steps: ["bundle install"]) do |x| + x["label"] += " [rack-2]" + x["env"]["RACK"] = "~> 2.0" +end + +step_for("railties", "test", service: "railties", pre_steps: ["bundle install"]) do |x| + x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") + x["label"] += " [rack-2]" + x["env"]["RACK"] = "~> 2.0" +end + +step_for("actionpack", "test", service: "default", pre_steps: ["rm Gemfile.lock", "bundle install"]) do |x| + x["label"] += " [rack-head]" + x["env"]["RACK"] = "head" + x["soft_fail"] = true +end + +step_for("railties", "test", service: "railties", pre_steps: ["rm Gemfile.lock", "bundle install"]) do |x| + x["parallelism"] = 12 if REPO_ROOT.join("railties/Rakefile").read.include?("BUILDKITE_PARALLEL") + x["label"] += " [rack-head]" + x["env"]["RACK"] = "head" + x["soft_fail"] = true +end + +# Ugly hacks to just get the build passing for now +STEPS.find { |s| s["label"] == "activestorage (2.2)" }&.tap do |s| + s["soft_fail"] = true +end +if RAILS_VERSION < Gem::Version.new("7.x") && RAILS_VERSION >= Gem::Version.new("6.1") + STEPS.delete_if { |s| s["label"] == "guides (2.7)" || s["label"] == "guides (3.0)" } +end +STEPS.delete_if { |s| s["label"] =~ /^guides/ } if RAILS_VERSION < Gem::Version.new("7.0") + +### + +STEPS.sort_by! do |step| + [ + -step["timeout_in_minutes"], + step["group"] == "isolated" ? 2 : 1, + step["command"].include?("test:") ? 2 : 1, + step["label"], + ] +end + +groups = STEPS.group_by { |s| s.delete("group") }.map do |group, steps| + { "group" => group, "steps" => steps } +end + +Buildkite::Builder.pipeline do + group do + label "build" + (RUBIES - [YJIT_RUBY]).map do |ruby| + command do + label ":docker: #{ruby}" + key "docker-image-#{ruby.gsub(/\W/, "-")}" + plugin ARTIFACTS_PLUGIN, { + "download" => [".dockerignore", ".buildkite/*", ".buildkite/**/*"], + } + + plugin DOCKER_COMPOSE_PLUGIN, { + "build" => "base", + "config" => ".buildkite/docker-compose.yml", + "env" => [ + "PRE_STEPS", + "RACK" + ], + "image-name" => image_name_for(ruby, short: true), + "cache-from" => [ + REBUILD_ID && "base:" + image_name_for(ruby, REBUILD_ID), + PULL_REQUEST && "base:" + image_name_for(ruby, "pr-#{PULL_REQUEST}"), + LOCAL_BRANCH && LOCAL_BRANCH !~ /:/ && "base:" + image_name_for(ruby, "br-#{LOCAL_BRANCH}"), + BASE_BRANCH && "base:" + image_name_for(ruby, "br-#{BASE_BRANCH}"), + "base:" + image_name_for(ruby, "br-main"), + ].grep(String).uniq, + "push" => [ + LOCAL_BRANCH =~ /:/ ? + "base:" + image_name_for(ruby, "pr-#{PULL_REQUEST}") : + "base:" + image_name_for(ruby, "br-#{LOCAL_BRANCH}"), + ], + "image-repository" => IMAGE_BASE, + } + + env({ + RUBY_IMAGE: ruby_image(ruby), + encrypted_0fb9444d0374_key: nil, + encrypted_0fb9444d0374_iv: nil, + }) + + timeout_in_minutes 15 + if SOFT_FAIL.include?(ruby) + soft_fail true + end + agents queue: BUILD_QUEUE + end + end + end + + groups.map do |_group| + group do + label _group["group"] + + _group["steps"].map do |_step| + command do + label _step["label"] + depends_on _step["depends_on"] + command _step["command"] + + #plugin ARTIFACTS_PLUGIN, { + # "download" => [".buildkite/*", ".buildkite/**/*"], + #}, + + plugins _step["plugins"] + env _step["env"] + timeout_in_minutes _step["timeout_in_minutes"] + + if _step["soft_fail"] + soft_fail true + end + + agents _step["agents"] + artifact_paths _step["artifact_paths"] + automatic_retry_on exit_status: -1, limit: 2 + end + end + end + end +end \ No newline at end of file diff --git a/rails-initial-pipeline.yml b/rails-initial-pipeline.yml index 32dfe416..74fb2087 100644 --- a/rails-initial-pipeline.yml +++ b/rails-initial-pipeline.yml @@ -2,7 +2,7 @@ # configuration in the Buildkite UI. steps: - - name: ":pipeline:" + - name: ":pipeline: rails-initial-pipeline" command: | PATH=/bin:/usr/bin set -e @@ -22,6 +22,8 @@ steps: GIT_BRANCH="$${CONFIG_BRANCH-main}" GIT_BRANCH="$${GIT_BRANCH#*:}" + echo "Cloning buildkite-config:" + echo "git clone -b \"$$GIT_BRANCH\" \"$$GIT_REPO\" .buildkite" git clone -b "$$GIT_BRANCH" "$$GIT_REPO" .buildkite rm -rf .buildkite/.git @@ -66,6 +68,7 @@ steps: -e BUILDKITE_AGENT_META_DATA_QUEUE -e BUILDKITE_BRANCH -e BUILDKITE_BUILD_ID + -e BUILDKITE_PIPELINE_NAME -e BUILDKITE_PULL_REQUEST -e BUILDKITE_PULL_REQUEST_BASE_BRANCH -e BUILDKITE_REBUILT_FROM_BUILD_ID