Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 2 additions & 3 deletions lib/mars/aggregator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,11 @@

module MARS
class Aggregator < Runnable
attr_reader :name, :operation
attr_reader :operation

def initialize(name = "Aggregator", operation: nil, **kwargs)
super(**kwargs)
super(name: name, **kwargs)

@name = name
@operation = operation || ->(inputs) { inputs }
end

Expand Down
5 changes: 1 addition & 4 deletions lib/mars/gate.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,9 @@

module MARS
class Gate < Runnable
attr_reader :name

def initialize(name = "Gate", condition:, branches:, **kwargs)
super(**kwargs)
super(name: name, **kwargs)

@name = name
@condition = condition
@branches = branches
end
Expand Down
20 changes: 19 additions & 1 deletion lib/mars/runnable.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,28 @@

module MARS
class Runnable
include Hooks

attr_reader :name, :formatter
attr_accessor :state

def initialize(state: {})
class << self
def step_name
return @step_name if defined?(@step_name)
return unless name

name.split("::").last.gsub(/([a-z])([A-Z])/, '\1_\2').downcase
end

def formatter(klass = nil)
klass ? @formatter_class = klass : @formatter_class
end
end

def initialize(name: self.class.step_name, state: {}, formatter: nil)
@name = name
@state = state
@formatter = formatter || self.class.formatter&.new || Formatter.new
end

def run(input)
Expand Down
5 changes: 1 addition & 4 deletions lib/mars/workflows/parallel.rb
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,9 @@
module MARS
module Workflows
class Parallel < Runnable
attr_reader :name

def initialize(name, steps:, aggregator: nil, **kwargs)
super(**kwargs)
super(name: name, **kwargs)

@name = name
@steps = steps
@aggregator = aggregator || Aggregator.new("#{name} Aggregator")
end
Expand Down
5 changes: 1 addition & 4 deletions lib/mars/workflows/sequential.rb
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,9 @@
module MARS
module Workflows
class Sequential < Runnable
attr_reader :name

def initialize(name, steps:, **kwargs)
super(**kwargs)
super(name: name, **kwargs)

@name = name
@steps = steps
end

Expand Down
67 changes: 67 additions & 0 deletions spec/mars/runnable_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,73 @@ def run(input)
end
end

describe "#name" do
it "defaults to nil for anonymous classes" do
klass = Class.new(described_class)
expect(klass.new.name).to be_nil
end

it "can be set via the name keyword" do
runnable = described_class.new(name: "my_step")
expect(runnable.name).to eq("my_step")
end

it "derives step_name from the class name" do
stub_const("MARS::MyCustomStep", Class.new(described_class))
expect(MARS::MyCustomStep.new.name).to eq("my_custom_step")
end
end

describe "#formatter" do
it "defaults to a Formatter instance" do
runnable = described_class.new
expect(runnable.formatter).to be_a(MARS::Formatter)
end

it "can be set via the formatter keyword" do
custom_formatter = MARS::Formatter.new
runnable = described_class.new(formatter: custom_formatter)
expect(runnable.formatter).to eq(custom_formatter)
end

it "uses the class-level formatter when declared" do
custom_formatter_class = Class.new(MARS::Formatter)
klass = Class.new(described_class) do
formatter custom_formatter_class
end

expect(klass.new.formatter).to be_a(custom_formatter_class)
end
end

describe "hooks" do
it "includes Hooks module" do
expect(described_class.ancestors).to include(MARS::Hooks)
end

it "supports before_run hooks" do
klass = Class.new(described_class)
calls = []
klass.before_run { |_ctx, step| calls << step.name }

step = klass.new(name: "test")
step.run_before_hooks(MARS::ExecutionContext.new(input: "x"))

expect(calls).to eq(["test"])
end

it "supports after_run hooks" do
klass = Class.new(described_class)
calls = []
klass.after_run { |_ctx, result, _step| calls << result }

step = klass.new(name: "test")
step.run_after_hooks(MARS::ExecutionContext.new(input: "x"), "result")

expect(calls).to eq(["result"])
end
end

describe "inheritance" do
it "can be inherited" do
subclass = Class.new(described_class)
Expand Down