Andrew Kane Gem Writer
Andrew Kane Gem Writer
Section titled “Andrew Kane Gem Writer”Write Ruby gems following Andrew Kane’s battle-tested patterns from 100+ gems with 374M+ downloads (Searchkick, PgHero, Chartkick, Strong Migrations, Lockbox, Ahoy, Blazer, Groupdate, Neighbor, Blind Index).
Core Philosophy
Section titled “Core Philosophy”Simplicity over cleverness. Zero or minimal dependencies. Explicit code over metaprogramming. Rails integration without Rails coupling. Every pattern serves production use cases.
Entry Point Structure
Section titled “Entry Point Structure”Every gem follows this exact pattern in lib/gemname.rb:
# 1. Dependencies (stdlib preferred)require "forwardable"
# 2. Internal modulesrequire_relative "gemname/model"require_relative "gemname/version"
# 3. Conditional Rails (CRITICAL - never require Rails directly)require_relative "gemname/railtie" if defined?(Rails)
# 4. Module with config and errorsmodule GemName class Error < StandardError; end class InvalidConfigError < Error; end
class << self attr_accessor :timeout, :logger attr_writer :client end
self.timeout = 10 # Defaults set immediatelyendClass Macro DSL Pattern
Section titled “Class Macro DSL Pattern”The signature Kane pattern—single method call configures everything:
# Usageclass Product < ApplicationRecord searchkick word_start: [:name]end
# Implementationmodule GemName module Model def gemname(**options) unknown = options.keys - KNOWN_KEYWORDS raise ArgumentError, "unknown keywords: #{unknown.join(", ")}" if unknown.any?
mod = Module.new mod.module_eval do define_method :some_method do # implementation end unless method_defined?(:some_method) end include mod
class_eval do cattr_reader :gemname_options, instance_reader: false class_variable_set :@@gemname_options, options.dup end end endendRails Integration
Section titled “Rails Integration”Always use ActiveSupport.on_load—never require Rails gems directly:
# WRONGrequire "active_record"ActiveRecord::Base.include(MyGem::Model)
# CORRECTActiveSupport.on_load(:active_record) do extend GemName::Modelend
# Use prepend for behavior modificationActiveSupport.on_load(:active_record) do ActiveRecord::Migration.prepend(GemName::Migration)endConfiguration Pattern
Section titled “Configuration Pattern”Use class << self with attr_accessor, not Configuration objects:
module GemName class << self attr_accessor :timeout, :logger attr_writer :master_key end
def self.master_key @master_key ||= ENV["GEMNAME_MASTER_KEY"] end
self.timeout = 10 self.logger = nilendError Handling
Section titled “Error Handling”Simple hierarchy with informative messages:
module GemName class Error < StandardError; end class ConfigError < Error; end class ValidationError < Error; endend
# Validate early with ArgumentErrordef initialize(key:) raise ArgumentError, "Key must be 32 bytes" unless key&.bytesize == 32endTesting (Minitest Only)
Section titled “Testing (Minitest Only)”require "bundler/setup"Bundler.require(:default)require "minitest/autorun"require "minitest/pride"
# test/model_test.rbclass ModelTest < Minitest::Test def test_basic_functionality assert_equal expected, actual endendGemspec Pattern
Section titled “Gemspec Pattern”Zero runtime dependencies when possible:
Gem::Specification.new do |spec| spec.name = "gemname" spec.version = GemName::VERSION spec.required_ruby_version = ">= 3.1" spec.files = Dir["*.{md,txt}", "{lib}/**/*"] spec.require_path = "lib" # NO add_dependency lines - dev deps go in GemfileendAnti-Patterns to Avoid
Section titled “Anti-Patterns to Avoid”method_missing(usedefine_methodinstead)- Configuration objects (use class accessors)
@@class_variables(useclass << self)- Requiring Rails gems directly
- Many runtime dependencies
- Committing Gemfile.lock in gems
- RSpec (use Minitest)
- Heavy DSLs (prefer explicit Ruby)
Reference Files
Section titled “Reference Files”For deeper patterns, see:
- references/module-organization.md - Directory layouts, method decomposition
- references/rails-integration.md - Railtie, Engine, on_load patterns
- references/database-adapters.md - Multi-database support patterns
- references/testing-patterns.md - Multi-version testing, CI setup
- references/resources.md - Links to Kane’s repos and articles