3

我的应用程序的结构:

.
├── config
│   ├── boot.rb
│   └── environment.rb
├── Gemfile
├── Gemfile.lock
├── lib
│   ├── entities
│   │   └── account.rb
│   └── repositories
│       └── account_repository.rb
└── README.md

宝石文件:

source 'https://rubygems.org'

gem 'pg', '~> 1.1'
gem 'dotenv'
gem 'byebug'
gem 'hanami-model'

配置/环境.rb:

require 'bundler/setup'
require 'hanami/model'
require 'dotenv/load'

class App
  class << self
    def boot
      Mutex.new.synchronize do
        Hanami::Model.configure do
          adapter :sql, ENV['DATABASE_URL']
        end.load!
      end
    end
  end
end

配置/boot.rb:

require_relative './environment'
App.boot

lib/entities/account.rb:

require 'hanami/model'
require_relative '../repositories/account_repository'

class Account < Hanami::Entity
end

lib/repositories/account_repository.rb:

require 'hanami/model'
require_relative '../entities/account'

class AccountRepository < Hanami::Repository
   self.relation = :accounts
end

在控制台中,我运行以下代码,但出现错误:

irb -I .
irb(main):001:0> require 'config/boot'
=> true
irb(main):002:0> require 'lib/repositories/account_repository'
=> true
irb(main):003:0> rep = AccountRepository.new
Traceback (most recent call last):
        6: from /home/mvalitov/.asdf/installs/ruby/2.5.1/bin/irb:11:in `<main>'
        5: from (irb):3
        4: from (irb):3:in `new'
        3: from /home/mvalitov/.asdf/installs/ruby/2.5.1/lib/ruby/gems/2.5.0/gems/hanami-model-1.3.2/lib/hanami/repository.rb:420:in `initialize'
        2: from /home/mvalitov/.asdf/installs/ruby/2.5.1/lib/ruby/gems/2.5.0/gems/rom-repository-1.4.0/lib/rom/repository/root.rb:62:in `initialize'
        1: from /home/mvalitov/.asdf/installs/ruby/2.5.1/lib/ruby/gems/2.5.0/gems/rom-3.3.3/lib/rom/registry.rb:30:in `fetch'
ArgumentError (key cannot be nil)

我究竟做错了什么?如果将所有实体代码和存储库放在一个文件中,则代码运行时不会出错。

4

2 回答 2

1

问题在于加载顺序。

以下是文档中关于严格顺序的说明:

使用 sql 适配器时,必须在 Hanami::Model.load 之前需要 hanami/model/sql!被调用以便正确加载关系。

来源:https ://github.com/hanami/model#mapping

因此,对于您的情况,需要App.boot在所有声明之后运行。

详细说明:

如果您将所有代码放入一个文件中,您将看到不同之处:

# run.rb
require 'bundler/setup'
require 'hanami/model'
require 'dotenv/load'
class App
  class << self
    def boot
      Mutex.new.synchronize do
        Hanami::Model.configure do
          adapter :sql, 'postgresql://postgres:12345@localhost:5432/mame-challenge_development'
          path '/home/mifrill/Documents/source/hamani-bug'
        end.load!
      end
    end
  end
end
App.boot
class AccountRepository < Hanami::Repository
  self.relation = :accounts
end
class Account < Hanami::Entity
end
AccountRepository.new

ruby run.rb

gems/rom-3.3.3/lib/rom/registry.rb:30:in `fetch': key cannot be nil (ArgumentError)

App.boot在存储库和实体定义之后移动,如下所示:

require 'bundler/setup'
require 'hanami/model'
require 'dotenv/load'
class App
  class << self
    def boot
      Mutex.new.synchronize do
        Hanami::Model.configure do
          adapter :sql, 'postgresql://postgres:12345@localhost:5432/mame-challenge_development'
          path '/home/mifrill/Documents/source/hamani-bug'
        end.load!
      end
    end
  end
end
class AccountRepository < Hanami::Repository
  self.relation = :accounts
end
class Account < Hanami::Entity
end
App.boot
AccountRepository.new

ruby run.rb

{:accounts=>#<ROM::Relation[Accounts] name=accounts dataset=#<Sequel::Postgres::Dataset: "SELECT * FROM \"accounts\"">>

因此,尝试在load!解决之前要求存储库文件:

config/boot.rb

require_relative './environment'
require_relative '../lib/repositories/account_repository'
App.boot
irb -I .
require 'config/boot'
rep = AccountRepository.new
...
{:accounts=>#<ROM::Relation[Accounts] name=accounts dataset=#<Sequel::Postgres::Dataset: "SELECT * FROM \"accounts\"">>}

于 2021-10-30T17:38:30.050 回答
0

您缺少“根”配置/选项(我不太了解,因为我不使用 Hanami)。

我是怎么找到的?

查看堆栈跟踪:

(所有链接都是最新的,因为我很懒:))

于 2019-03-22T12:32:21.290 回答