# frozen_string_literal: true RSpec.describe Dry::Transformer::Function do let(:container) do Module.new do extend Dry::Transformer::Registry import Dry::Transformer::HashTransformations end end describe '#name' do let(:block) { proc { |v| v } } it 'returns the name of the module function' do expect(container[:symbolize_keys].name).to eql :symbolize_keys end it 'returns the explicitly assigned name' do expect(described_class.new(block, name: :identity).name).to eql :identity end it 'returns the unnamed closure' do expect(described_class.new(block).name).to eql block end end describe '#>>' do it 'composes named functions' do f1 = container[:symbolize_keys] f2 = container[:rename_keys, user_name: :name] f3 = f1 >> f2 expect(f3.to_ast).to eql( [ :symbolize_keys, [], [ :rename_keys, [user_name: :name] ] ] ) expect(f3['user_name' => 'Jane']).to eql(name: 'Jane') f4 = f3 >> container[:nest, :details, [:name]] expect(f4.to_ast).to eql( [ :symbolize_keys, [], [ :rename_keys, [user_name: :name] ], [ :nest, [:details, [:name]] ] ] ) expect(f4['user_name' => 'Jane']).to eql(details: { name: 'Jane' }) end it 'composes anonymous functions' do f1 = container[->(v, m) { v * m }, 2] f2 = container[:to_s.to_proc] f3 = f1 >> f2 expect(f3.to_ast).to eql( [ f1.fn, [2], [ f2.fn, [] ] ] ) end it 'plays well with registered compositions' do container.register(:user_names, container[:symbolize_keys] + container[:rename_keys, user_name: :name]) f = container[:user_names] expect(f['user_name' => 'Jane']).to eql(name: 'Jane') expect(f.to_ast).to eql( [ :symbolize_keys, [], [ :rename_keys, [user_name: :name] ] ] ) end it 'plays well with registered functions' do container.register(:to_string, Dry::Transformer::Coercions.t(:to_string)) fn = container.t(:to_string) expect(fn[:ok]).to eql('ok') expect(fn.to_ast).to eql([:to_string, []]) end it 'plays well with functions as arguments' do container.register(:map_array, Dry::Transformer::ArrayTransformations.t(:map_array)) container.register(:to_symbol, Dry::Transformer::Coercions.t(:to_symbol)) fn = container.t(:map_array, container.t(:to_symbol)) expect(fn.call(%w(a b c))).to eql([:a, :b, :c]) expect(fn.to_ast).to eql( [ :map_array, [ [:to_symbol, []] ] ] ) end end describe '#==' do let(:fns) do Module.new do extend Dry::Transformer::Registry import :wrap, from: Dry::Transformer::ArrayTransformations end end it 'returns true when the other is equal' do left = fns[:wrap, :user, [:name, :email]] right = fns[:wrap, :user, [:name, :email]] expect(left == right).to be(true) end it 'returns false when the other is not a fn' do left = fns[:wrap, :user, [:name, :email]] right = 'boo!' expect(left == right).to be(false) end end describe '#to_proc' do shared_examples :providing_a_proc do let(:fn) { described_class.new(source) } subject { fn.to_proc } it 'returns a proc' do expect(subject).to be_instance_of Proc end it 'works fine' do expect(subject.call :foo).to eql('foo') end end context 'from a method' do let(:source) do mod = Module.new do def self.get(x) x.to_s end end mod.method(:get) end it_behaves_like :providing_a_proc end context 'from a proc' do let(:source) { -> value { value.to_s } } it_behaves_like :providing_a_proc end context 'from a transproc' do let(:source) { Dry::Transformer::Function.new -> value { value.to_s } } it_behaves_like :providing_a_proc it 'can be applied to collection' do expect([:foo, :bar].map(&source)).to eql(%w(foo bar)) end end context 'with curried args' do let(:source) { -> i, j { [i, j].join(' ') } } it 'works fine' do fn = described_class.new(source, args: ['world']) result = fn.to_proc.call('hello') expect(result).to eql('hello world') end end end end