概要
オニオンアーキテクチャはJeffrey Palermo氏により考案されたアーキテクチャパターンである。伝統的な階層化アーキテクチャとオブジェクト指向の考え方を踏襲しつつ、これまでよりも保守性、テスト容易性、依存性の点で優れたアプリケーションを構築することを目的としている。本記事ではこのオニオンアーキテクチャとは何かについてPalermo氏の記事を参考にして考察する。
前提となる知識
オニオンアーキテクチャを理解する上で前提となるいくつかの知識を挙げる。
オブジェクト指向
オニオンアーキテクチャはオブジェクト指向を前提としたアーキテクチャである。
階層化アーキテクチャ
オニオンアーキテクチャは多層アーキテクチャを利用している。
依存性逆転の原則
オニオンアーキテクチャでは依存性逆転の原則がオブジェクトのモデリングに利用されている。
ドメイン駆動設計
Entity, Repositoryといったドメイン駆動設計で提唱されているモデリングの考え方はオニオンアーキテクチャでも一部利用されている。
伝統的な階層化アーキテクチャが抱える課題
Jeffrey Palermo氏はオニオンアーキテクチャを説明する前に、伝統的な階層化アーキテクチャが抱える問題について言及している。
伝統的な階層化アーキテクチャの一例として、プレゼンテーション層(もしくはui層)、モデル層(もしくはビジネスロジック層)、データアクセス層、infrastructure層からなる階層化アーキテクチャを考えてみる。
このアーキテクチャを利用すると、各層にあるロジックは通常その下の層に対して依存していく。加えて、基本的にはプレゼンテーション層、モデル層、データ層からinfrastructure層(File Access, DB Access, ORM, etc...)に対する依存が発生するため、全ての層がinfrastructure層に対して密結合になる可能性がある。infrastructure層にあるロジックに修正が発生した場合は、そのロジックを参照している全ての層のロジックも修正する必要がある。
オニオンアーキテクチャの階層構造
オニオンアーキテクチャでは階層構造を円で表現する。オニオン(=玉ねぎ)という名称は、おそらくこの円構造からきているものと思われる。
全ての依存関係は円の中心の層に対して向かう。一方で、中心の層から外側の層へは依存しない。アプリケーションのコア(核)になる層の数は変化しても良いが、ドメインモデル層は常に中心に配置する。
Jeffrey Palermo氏の説明では階層構造が円で表現されているが、下記のように表現しても問題ないだろう。
各層の役割
オニオンアーキテクチャの各層の役割について説明する。
ドメインモデル層
ビジネスロジックに関連した状態と振る舞いの一体化したオブジェクトを配置する。ドメインモデル層のロジックは、他の層への依存関係を持たない。
E.g.
Entity, ValueObjectなど
ドメインサービス層
ビジネスロジックに関わる振る舞いのロジック、interfaceなどを配置する。
E.g.
DomainService, IRepositoryなど
アプリケーションサービス層
アプリケーション固有のロジック、一般的によく利用される制御用のinterafaceなどを配置する。
E.g.
ApplicationService, IUserSessionなど
UI, Infrastructure層
伝統的な階層化アーキテクチャで表現されるinfrastructure層のオブジェクト(File Access, DB Access, ORM, etc...)、ドメインサービス層で用意したIRepositoryの実態, MVC, やユニットテストなどをここに配置する。
オニオンアーキテクチャを利用したクラス設計の例
上述の各層の役割を前提とし、簡易なクラス設計の例を下記に示す。
Controller、UserSession、Repository、FileAccessorといったオブジェクトはJeffrey Palermo氏の説明を参考に、EntityとVOといったオブジェクトはRobert Martin氏の説明やEric Evans氏の説明を参考にしている。
オニオンアーキテクチャの教義
Jeffrey Palermo氏によると、下記がオニオンアーキテクチャの教義として挙げられている。
- アプリケーションのロジックは独立したドメインモデルを取り囲むように配置される。
- infrastructure層と別の内側の層にinterfaceを定義する。interfaceを定義した層とは別の外側の層がinterfaceを実装する。
- (依存性逆転の原則を利用しているため)依存の方向は常に外側の層から内側の層に向かう。
- application coreのロジック(*注1)はinfrastructure層から切り離される。
伝統的な階層化アーキテクチャとの違い
Jeffrey Palermo氏によると、オニオンアーキテクチャと伝統的な階層化アーキテクチャの違いは、階層間の呼び出し方や依存関係にあると説明されている。
階層間の依存関係の違い
伝統的な階層化アーキテクチャでは、基本的に一つ下の階層に依存する。UI層がData Access層に依存することはない。(*注2)
一方オニオンアーキテクチャでは、外側の階層から内側の階層に依存する時、一つ下以外の内側の階層に対して依存することが可能である。
円形による依存関係の表現
https://qiita.com/cocoa-maemae/items/e3f2eabbe0877c2af8d0#伝統的な階層化アーキテクチャが抱える課題
で説明したように、伝統的な階層化アーキテクチャでは、ビジネスロジックやデータアクセス層とinfrastrcture層の依存関係が密になってしまう特徴がある。
一方オニオンアーキテクチャでは、application coreからinfrastructure層のロジックを呼び出す際常にinterfaceを経由するため、infrastructure層の変更がビジネスロジック(内側の層)に影響を与えることがなくなる。よってビジネスロジックとinfrastrcture層の依存関係は伝統的な階層化アーキテクチャよりも疎になる。
まとめ
本記事ではオニオンアーキテクチャについて、階層構造、クラス設計、教義といった観点から説明してきた。オニオンアーキテクチャで新規性があるのは、依存性逆転の法則を利用してinfrastructure層とui(presentation)層を同じ階層位置として捉える点(*注3)と、ビジネスロジックを配置するdomain層とinfrastructure層の依存関係を疎にする点にあると思われる。この考え方は、オニオンアーキテクチャよりも後に提唱され、日本でも大ヒットしたクリーンアーキテクチャにも取り入れられている。
(*注1)Jeffrey Palermo氏のブログによると、application coreはapplication service, domain service, domain modelの3層に分離されているが、それぞれの層がどんな役割を持っているかまでは明確に説明されていない。
(*注2)Jeffrey Palermo氏はこの様に主張しているが、従来型の階層化アーキテクチャでもUI層からData Access層(ORMなど)を直接呼ぶことは可能であるため、この主張には疑問が残る。
(*注3)
オニオンアーキテクチャよりも前に提唱されたヘキサゴナルアーキテクチャも同じようにinfrastructure層とui(presentation)層を対称的な関係として捉えていた。
参考
- https://jeffreypalermo.com/2008/07/the-onion-architecture-part-1/
- https://jeffreypalermo.com/2008/08/the-onion-architecture-part-3/
- https://www.codeguru.com/csharp/csharp/cs_misc/designtechniques/understanding-onion-architecture.html
- https://tech.ovoenergy.com/onion-architecture/
- https://medium.com/@sergiis/stem-in-onion-architecture-or-fallacy-of-data-layer-9923f398f215
- https://blog.tai2.net/hexagonal_architexture.html
- https://qiita.com/gki/items/91386b082c57123f1ba0