diff --git a/Dockerfile b/Dockerfile index 8467a7f..dea4631 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,11 +1,7 @@ -FROM ruby:3.2-alpine -LABEL maintainer=engineering@kodeco.com +ARG RUBY_ENV -LABEL com.github.actions.name="robles" -LABEL com.github.actions.author="Kodeco " -LABEL com.github.actions.description="Content publication for kodeco.com" -LABEL com.github.actions.color="purple" -LABEL com.github.actions.icon="book" +FROM ruby:3.2-alpine AS builder +LABEL maintainer=engineering@kodeco.com ARG APP_ROOT=/app/robles ARG BUILD_PACKAGES="build-base git" @@ -41,3 +37,52 @@ RUN bundle install --jobs 20 --retry 5 # Copy the main application. COPY . ./ + +# Remove extra files +RUN rm -rf /usr/local/bundle/cache/* + + +############################## +# PACKAGE STAGE # +############################## +FROM ruby:3.2-alpine +LABEL maintainer=engineering@kodeco.com +LABEL com.github.actions.name="robles" +LABEL com.github.actions.author="Kodeco " +LABEL com.github.actions.description="Content publication for kodeco.com" +LABEL com.github.actions.color="purple" +LABEL com.github.actions.icon="book" + +ARG APP_ROOT=/app/robles +ARG RUBY_ENV=${RUBY_ENV:-production} + +ENV RUBY_ENV=${RUBY_ENV} + +ARG RUNTIME_PACKAGES="imagemagick git tzdata" +ARG TEST_AND_DEV_PACKAGES="bash build-base libsodium-dev" + +# SYSLOG TO STDOUT +RUN \ + touch /var/log/syslog && \ + ln -sf /proc/1/fd/1 /var/log/syslog + +# libsodium +COPY --from=builder /usr/lib/libsodium.so* /usr/lib/ +# Copy the app from builder +COPY --from=builder $APP_ROOT $APP_ROOT +COPY --from=builder /usr/local/bundle /usr/local/bundle + +# Set the working directory +WORKDIR $APP_ROOT + +# For runtime +RUN apk update \ + && apk upgrade \ + && apk add --update --no-cache $RUNTIME_PACKAGES \ + && rm -rf /var/cache/apk/* + +# Test and dev packages +RUN if [ "$RUBY_ENV" == "test" -o "$RUBY_ENV" == "development" ]; then \ + apk add --update --no-cache $TEST_AND_DEV_PACKAGES \ + && rm -rf /var/cache/apk/*; \ + fi diff --git a/Gemfile b/Gemfile index 272f28c..63f86cb 100644 --- a/Gemfile +++ b/Gemfile @@ -13,7 +13,9 @@ gem 'cli-ui', '~> 2' gem 'thor', '~> 1.0', '>= 1.0.1' # Markdown processing -gem 'commonmarker' +# >= 1 switches out the underlying library to one that does not support musl +# Currently, there are no plans to support it, so let's lock to < 1 +gem 'commonmarker', '< 1' # HTTP Client gem 'faraday', '~> 2' diff --git a/Gemfile.lock b/Gemfile.lock index ad4060b..34f7fbf 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -14,9 +14,9 @@ GIT GEM remote: https://rubygems.org/ specs: - activemodel (7.1.1) - activesupport (= 7.1.1) - activesupport (7.1.1) + activemodel (7.1.2) + activesupport (= 7.1.2) + activesupport (7.1.2) base64 bigdecimal concurrent-ruby (~> 1.0, >= 1.0.2) @@ -26,29 +26,29 @@ GEM minitest (>= 5.1) mutex_m tzinfo (~> 2.0) - addressable (2.8.5) + addressable (2.8.6) public_suffix (>= 2.0.2, < 6.0) ast (2.4.2) - aws-eventstream (1.2.0) - aws-partitions (1.835.0) - aws-sdk-core (3.185.1) - aws-eventstream (~> 1, >= 1.0.2) + aws-eventstream (1.3.0) + aws-partitions (1.877.0) + aws-sdk-core (3.190.1) + aws-eventstream (~> 1, >= 1.3.0) aws-partitions (~> 1, >= 1.651.0) - aws-sigv4 (~> 1.5) + aws-sigv4 (~> 1.8) jmespath (~> 1, >= 1.6.1) - aws-sdk-kms (1.72.0) - aws-sdk-core (~> 3, >= 3.184.0) + aws-sdk-kms (1.76.0) + aws-sdk-core (~> 3, >= 3.188.0) aws-sigv4 (~> 1.1) - aws-sdk-s3 (1.136.0) - aws-sdk-core (~> 3, >= 3.181.0) + aws-sdk-s3 (1.142.0) + aws-sdk-core (~> 3, >= 3.189.0) aws-sdk-kms (~> 1) - aws-sigv4 (~> 1.6) - aws-sigv4 (1.6.0) + aws-sigv4 (~> 1.8) + aws-sigv4 (1.8.0) aws-eventstream (~> 1, >= 1.0.2) backport (1.2.0) - base64 (0.1.1) - benchmark (0.2.1) - bigdecimal (3.1.4) + base64 (0.2.0) + benchmark (0.3.0) + bigdecimal (3.1.5) cli-ui (2.2.3) coderay (1.1.3) commonmarker (0.23.10) @@ -56,14 +56,14 @@ GEM connection_pool (2.4.1) daemons (1.4.1) diff-lcs (1.5.0) - drb (2.1.1) + drb (2.2.0) ruby2_keywords e2mmap (0.1.0) em-websocket (0.5.3) eventmachine (>= 0.12.9) http_parser.rb (~> 0) eventmachine (1.2.7) - faraday (2.7.11) + faraday (2.8.1) base64 faraday-net_http (>= 2.0, < 3.1) ruby2_keywords (>= 0.0.4) @@ -77,7 +77,7 @@ GEM websocket-driver (>= 0.6, < 0.8) ffi (1.16.3) formatador (1.1.0) - git (1.18.0) + git (1.19.0) addressable (~> 2.8) rchardet (~> 1.8) google-protobuf (3.22.0) @@ -101,7 +101,7 @@ GEM concurrent-ruby (~> 1.0) jaro_winkler (1.5.6) jmespath (1.6.2) - json (2.6.3) + json (2.7.1) kramdown (2.4.0) rexml kramdown-parser-gfm (1.1.0) @@ -110,17 +110,17 @@ GEM listen (3.8.0) rb-fsevent (~> 0.10, >= 0.10.3) rb-inotify (~> 0.9, >= 0.9.10) - lumberjack (1.2.9) + lumberjack (1.2.10) method_source (1.0.0) mini_magick (4.12.0) - mini_portile2 (2.8.4) + mini_portile2 (2.8.5) minitest (5.20.0) multi_json (1.15.0) mustermann (3.0.0) ruby2_keywords (~> 0.0.1) - mutex_m (0.1.2) + mutex_m (0.2.0) nenv (0.3.0) - nokogiri (1.15.4) + nokogiri (1.16.0) mini_portile2 (~> 2.8.2) racc (~> 1.4) notiffany (0.1.3) @@ -129,24 +129,25 @@ GEM octokit (7.2.0) faraday (>= 1, < 3) sawyer (~> 0.9) - parallel (1.23.0) - parser (3.2.2.4) + parallel (1.24.0) + parser (3.3.0.2) ast (~> 2.4.1) racc pry (0.14.2) coderay (~> 1.1) method_source (~> 1.0) - public_suffix (5.0.3) - racc (1.7.1) + public_suffix (5.0.4) + racc (1.7.3) rack (2.2.8) rack-livereload (0.5.1) rack - rack-protection (3.1.0) + rack-protection (3.2.0) + base64 (>= 0.1.0) rack (~> 2.2, >= 2.2.4) rack-test (2.1.0) rack (>= 1.3) rainbow (3.1.1) - rake (13.0.6) + rake (13.1.0) rb-fsevent (0.11.2) rb-inotify (0.10.1) ffi (~> 1.0) @@ -154,12 +155,11 @@ GEM ffi rbs (2.8.4) rchardet (1.8.0) - regexp_parser (2.8.2) + regexp_parser (2.9.0) reverse_markdown (2.1.1) nokogiri rexml (3.2.6) - rubocop (1.57.1) - base64 (~> 0.1.1) + rubocop (1.59.0) json (~> 2.3) language_server-protocol (>= 3.17.0) parallel (~> 1.10) @@ -167,10 +167,10 @@ GEM rainbow (>= 2.2.2, < 4.0) regexp_parser (>= 1.8, < 3.0) rexml (>= 3.2.5, < 4.0) - rubocop-ast (>= 1.28.1, < 2.0) + rubocop-ast (>= 1.30.0, < 2.0) ruby-progressbar (~> 1.7) unicode-display_width (>= 2.4.0, < 3.0) - rubocop-ast (1.29.0) + rubocop-ast (1.30.0) parser (>= 3.2.1.0) ruby-progressbar (1.13.0) ruby2_keywords (0.0.5) @@ -181,13 +181,13 @@ GEM addressable (>= 2.3.5) faraday (>= 0.17.3, < 3) shellany (0.0.1) - sinatra (3.1.0) + sinatra (3.2.0) mustermann (~> 3.0) rack (~> 2.2, >= 2.2.4) - rack-protection (= 3.1.0) + rack-protection (= 3.2.0) tilt (~> 2.0) slack-notifier (2.4.0) - solargraph (0.49.0) + solargraph (0.50.0) backport (~> 1.2) benchmark bundler (~> 2.0) @@ -207,7 +207,7 @@ GEM daemons (~> 1.0, >= 1.0.9) eventmachine (~> 1.0, >= 1.0.4) rack (>= 1, < 3) - thor (1.2.2) + thor (1.3.0) tilt (2.3.0) tzinfo (2.0.6) concurrent-ruby (~> 1.0) @@ -227,7 +227,7 @@ DEPENDENCIES activesupport (< 7.2) aws-sdk-s3 (~> 1.64) cli-ui (~> 2) - commonmarker + commonmarker (< 1) concurrent-ruby (~> 1.1) faraday (~> 2) faraday-retry @@ -253,4 +253,4 @@ DEPENDENCIES zeitwerk (~> 2.3) BUNDLED WITH - 2.4.20 + 2.5.4 diff --git a/app/commands/content_module_cli.rb b/app/commands/content_module_cli.rb index 75926e8..46f6fe3 100644 --- a/app/commands/content_module_cli.rb +++ b/app/commands/content_module_cli.rb @@ -55,20 +55,6 @@ def slides snapshotter.generate end - - desc 'serve', 'starts local preview server' - option :dev, type: :boolean, desc: 'Run in development mode (watch robles files, not module files)' - def serve - fork do - if options[:dev] - Guard.start(no_interactions: true) - else - Guard.start(guardfile_contents: content_module_guardfile, watchdir: './', no_interactions: true) - end - end - RoblesContentModuleServer.run! - end - desc 'secrets [REPO]', 'configures a module repo with the necessary secrets' long_desc <<-LONGDESC `robles module secrets [REPO]` will upload the secrets requires to run robles on a diff --git a/app/commands/video_cli.rb b/app/commands/video_cli.rb index 495e6d1..ae3b991 100644 --- a/app/commands/video_cli.rb +++ b/app/commands/video_cli.rb @@ -6,7 +6,7 @@ class VideoCli < Thor option :'release-file', type: :string, desc: 'Location of the release.yaml file' option :local, type: :boolean def render - video_course = runner.render_video_course(release_file: options['release_file'], local: options['local']) + runner.render_video_course(release_file: options['release_file'], local: options['local']) end desc 'serve', 'starts local preview server' @@ -39,8 +39,8 @@ def upload desc 'lint [RELEASE_FILE]', 'runs a selection of linters on the video course' option :'release-file', type: :string, desc: 'Location of the release.yaml file' - method_option 'without-version': :boolean, aliases: '-e', default: false, desc: 'Run linting without git branch naming check' - method_option silent: :boolean, aliases: '-s', default: false, desc: 'Hide all output' + method_option :'without-version', type: :boolean, aliases: '-e', default: false, desc: 'Run linting without git branch naming check' + method_option :silent, type: :boolean, aliases: '-s', default: false, desc: 'Hide all output' def lint output = runner.lint_video_course(release_file: options['publish_file'], options:) exit 1 unless output.validated || ENVIRONMENT == 'staging' diff --git a/app/lib/linting/book_linter.rb b/app/lib/linting/book_linter.rb index 87cba65..96bfed9 100644 --- a/app/lib/linting/book_linter.rb +++ b/app/lib/linting/book_linter.rb @@ -117,7 +117,7 @@ def book parser.parse end rescue Parser::Error => e - line_number = (e.message.match(/at line (\d+)/)&.captures&.first&.to_i || 0) + 1 + line_number = e.message.match(/at line (\d+)/)&.captures&.first.to_i + 1 annotations.push( Annotation.new( absolute_path: e.file, diff --git a/app/lib/linting/content_module_linter.rb b/app/lib/linting/content_module_linter.rb index 8c96176..ec6e411 100644 --- a/app/lib/linting/content_module_linter.rb +++ b/app/lib/linting/content_module_linter.rb @@ -109,7 +109,7 @@ def content_module parser.parse end rescue Parser::Error => e - line_number = (e.message.match(/at line (\d+)/)&.captures&.first&.to_i || 0) + 1 + line_number = e.message.match(/at line (\d+)/)&.captures&.first.to_i + 1 annotations.push( Annotation.new( absolute_path: e.file, diff --git a/app/lib/linting/metadata/captions_file.rb b/app/lib/linting/metadata/captions_file.rb index a527aa2..7e05fed 100644 --- a/app/lib/linting/metadata/captions_file.rb +++ b/app/lib/linting/metadata/captions_file.rb @@ -40,13 +40,14 @@ def video_course_captions def module_captions caption_files = ModuleFile.new(file:, attributes:).file_path_list caption_files.map do |file| - captions_file = if yaml?(file) - load_yaml(File.read(file)).deep_symbolize_keys[:captions_file] - else - @markdown_metadata = nil - @path = file - markdown_metadata[:captions_file] - end + captions_file = + if yaml?(file) + load_yaml(File.read(file)).deep_symbolize_keys[:captions_file] + else + @markdown_metadata = nil + @path = file + markdown_metadata[:captions_file] + end next unless captions_file.present? [captions_file, file] diff --git a/app/lib/linting/vend_linter.rb b/app/lib/linting/vend_linter.rb index 531eb44..5615945 100644 --- a/app/lib/linting/vend_linter.rb +++ b/app/lib/linting/vend_linter.rb @@ -104,7 +104,7 @@ def load_file parser.parse end rescue Parser::Error => e - line_number = (e.message.match(/at line (\d+)/)&.captures&.first&.to_i || 0) + 1 + line_number = e.message.match(/at line (\d+)/)&.captures&.first.to_i + 1 @annotations.push( Annotation.new( absolute_path: e.file, diff --git a/app/lib/linting/video_course_linter.rb b/app/lib/linting/video_course_linter.rb index 37a7503..5b46b17 100644 --- a/app/lib/linting/video_course_linter.rb +++ b/app/lib/linting/video_course_linter.rb @@ -109,7 +109,7 @@ def video_course parser.parse end rescue Parser::Error => e - line_number = (e.message.match(/at line (\d+)/)&.captures&.first&.to_i || 0) + 1 + line_number = e.message.match(/at line (\d+)/)&.captures&.first.to_i + 1 annotations.push( Annotation.new( absolute_path: e.file, diff --git a/app/lib/parser/content_module.rb b/app/lib/parser/content_module.rb index c9166df..a9ecb9c 100644 --- a/app/lib/parser/content_module.rb +++ b/app/lib/parser/content_module.rb @@ -47,7 +47,7 @@ def parse_segments(segment_yaml_file) lesson[:segments].map do |segment| segment_path = "#{lesson_path}/#{segment[:path]}" - send("parse_#{segment[:type]}".to_sym, segment_path) + send(:"parse_#{segment[:type]}", segment_path) end end diff --git a/app/lib/renderer/rw_markdown_renderer.rb b/app/lib/renderer/rw_markdown_renderer.rb index 9e147a0..71807d7 100644 --- a/app/lib/renderer/rw_markdown_renderer.rb +++ b/app/lib/renderer/rw_markdown_renderer.rb @@ -15,7 +15,7 @@ def initialize(image_provider:, root_path:, options: :DEFAULT, extensions: []) @root_path = root_path end - def image(node) # rubocop:disable Metrics/MethodLength, Metrics/AbcSize, Metrics/PerceivedComplexity + def image(node) # rubocop:disable Metrics/MethodLength, Metrics/AbcSize, Metrics/PerceivedComplexity, Metrics/CyclomaticComplexity return super(node) if image_provider.blank? title = node.title.present? ? escape_html(node.title) : '' diff --git a/app/lib/sinatra/pablo.rb b/app/lib/sinatra/pablo.rb index 3224b70..3f9c372 100644 --- a/app/lib/sinatra/pablo.rb +++ b/app/lib/sinatra/pablo.rb @@ -58,7 +58,7 @@ def save_path(path:, dir:, response:) [^/.]+ \. ( - #{app.settings.export_extensions.join("|")} + #{app.settings.export_extensions.join('|')} ) $}x file_path = Pathname(File.join(dir, path)) diff --git a/app/models/concerns/image_attachable.rb b/app/models/concerns/image_attachable.rb index 338822a..76141ec 100644 --- a/app/models/concerns/image_attachable.rb +++ b/app/models/concerns/image_attachable.rb @@ -48,7 +48,7 @@ def image_attachment_loop(&block) representations = block.call(local_url) .filter { |r| attribute[:variants].include?(r.variant) } .uniq(&:variant) - send("#{attribute[:destination]}=".to_sym, representations) + send(:"#{attribute[:destination]}=", representations) end end end diff --git a/app/models/concerns/markdown_renderable.rb b/app/models/concerns/markdown_renderable.rb index 3c31f50..dd92611 100644 --- a/app/models/concerns/markdown_renderable.rb +++ b/app/models/concerns/markdown_renderable.rb @@ -32,7 +32,7 @@ def markdown_render_loop(&block) rendered = block.call(content, attribute[:file], vtt_file) rendered = wrap_render(rendered, attribute[:wrapper_class]) - send("#{attribute[:destination]}=".to_sym, rendered) + send(:"#{attribute[:destination]}=", rendered) end end diff --git a/app/server/robles_content_module_server.rb b/app/server/robles_content_module_server.rb index 7183d2e..60459e3 100644 --- a/app/server/robles_content_module_server.rb +++ b/app/server/robles_content_module_server.rb @@ -3,7 +3,7 @@ require 'rack-livereload' # A local preview server for robles -class RoblesContentModuleServer < Sinatra::Application +class RoblesContentModuleServer < Sinatra::Application # rubocop:disable Metrics/ClassLength set :bind, '0.0.0.0' set :views, "#{__dir__}/views" set :public_folder, "#{__dir__}/public" diff --git a/app/server/robles_video_server.rb b/app/server/robles_video_server.rb index 76a6d49..f5e9e59 100644 --- a/app/server/robles_video_server.rb +++ b/app/server/robles_video_server.rb @@ -3,7 +3,7 @@ require 'rack-livereload' # A local preview server for robles -class RoblesVideoServer < Sinatra::Application # rubocop:disable Metrics/ClassLength +class RoblesVideoServer < Sinatra::Application set :bind, '0.0.0.0' set :views, "#{__dir__}/views" set :public_folder, "#{__dir__}/public" @@ -11,7 +11,7 @@ class RoblesVideoServer < Sinatra::Application # rubocop:disable Metrics/ClassLe use Rack::LiveReload, host: 'localhost', source: :vendored - helpers do # rubocop:disable Metrics/BlockLength + helpers do def slide_path(episode) "/slides/#{episode.slug}" end diff --git a/docker-compose.yml b/docker-compose.yml index eddded9..f265bf1 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,7 +1,11 @@ version: '3.7' services: app: - build: . + build: + context: . + dockerfile: Dockerfile + args: + - RUBY_ENV=development volumes: - .:/app/robles # When working on videos @@ -16,6 +20,7 @@ services: env_file: .env environment: - IMAGES_CDN_HOST=assets.robles.kodeco.com + - RUBY_ENV=development ports: - "4567:4567" - "35729:35729"