DTO可以引用域模型的VO吗?

时间:2019-08-06 10:42:43

标签: collections domain-driven-design dto value-objects

问题

数据传输对象(DTO)可以引用域模型的值对象(VO)吗?

上下文

在我的域中,我有一个进口商,它从集合中导入集合。收集由进口商依赖的收集器由DTO构建。既然导入器和收集器都是我域的服务(接口),那么DTO可以引用域值对象,还是仅在处理集合(聚合的生成)时才坚持使用原语并将它们变成值对象?

收集器实现,其中将构建由域模型中的值对象构成的DTO

<?php
/**
 * i-MSCP Patcher plugin
 *
 * @author        Laurent Declercq <l.declercq@nuxwin.com>
 * @copyright (C) 2019 Laurent Declercq <l.declercq@nuxwin.com>
 * @license       i-MSCP License <https://www.i-mscp.net/license-agreement.html>
 */

/**
 * @noinspection
 * PhpUnhandledExceptionInspection
 * PhpDocMissingThrowsInspection
 */

declare(strict_types=1);

namespace iMSCP\Plugin\Patcher\Infrastructure\Domain\Service\Component\Importer;

use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentBuild;
use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentName;
use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentVersion;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\ComponentCollector;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\DTO\ComponentDTO;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\DTO\ComponentDTOCollection;
use iMSCP_Config_Handler_File as MergedConfig;
use iMSCP_Plugin_Manager as PluginManager;
use RuntimeException;
use Throwable;

/**
 * Class DefaultComponentCollector
 * @package iMSCP\Plugin\Patcher\Infrastructure\Domain\Service\Component\Importer
 */
class DefaultComponentCollector implements ComponentCollector
{
    /**
     * @var MergedConfig
     */
    private $mergedConfig;

    /**
     * @var PluginManager
     */
    private $pluginManager;

    /**
     * DefaultComponentCollector constructor.
     *
     * @param MergedConfig $mergedConfig
     * @param PluginManager $pluginManager
     */
    public function __construct(
        MergedConfig $mergedConfig, PluginManager $pluginManager
    )
    {
        $this->mergedConfig = $mergedConfig;
        $this->pluginManager = $pluginManager;
    }

    /**
     * @inheritDoc
     */
    public function collect(ComponentDTOCollection $collection): void
    {
        try {
            // Core
            $collection->add(new ComponentDTO(
                ComponentName::fromString('core'),
                ComponentVersion::fromString($this->mergedConfig['Version']),
                ComponentBuild::fromString($this->mergedConfig['Build'])
            ));

            // Plugins
            $this->collectComponentsFromCorePluginManager($collection);
        } catch (Throwable $e) {
            throw new RuntimeException(sprintf(
                "Couldn't collect list of components: %s", $e->getMessage()
            ), $e->getCode(), $e);
        }
    }

    /**
     * Collects components from the i-MSCP core plugin manager.
     *
     * @param ComponentDTOCollection $collection
     * @return void
     */
    private function collectComponentsFromCorePluginManager(
        ComponentDTOCollection $collection
    ): void
    {
        $pluginList = $this->pluginManager->pluginGetList('all', false);

        foreach ($pluginList as $pluginName) {
            $pluginInfo = $this->pluginManager
                ->pluginGetInfo($pluginName)
                ->toArray();

            $componentDTO = new ComponentDTO(
                ComponentName::fromString($pluginInfo['name']),
                ComponentVersion::fromString($pluginInfo['version']),
                ComponentBuild::fromString((string)$pluginInfo['build'])
            );

            $collection->add($componentDTO);
        }
    }
}

1 个答案:

答案 0 :(得分:4)

  

数据传输对象(DTO)可以引用域模型的值对象(VO)吗?

是的,但是您要对此非常小心。

data transfer object的核心是消息。为了使消息达到其目的,发送者和接收者都必须对其语义具有兼容的理解。对DTO模式进行不兼容的更改需要对接收方进行相应的更改。

在域模型中,值对象不是消息。它是结构化的信息,纯粹是当前模型的实现细节。如果我们要部署模型的新版本,该版本使用完全不同的值排列或其底层数据结构,则可以。

因此,使DTO(应该是稳定的)依赖于值对象(不能保证稳定)会为将来的问题创造机会。

如果您的价值观很稳定,那么风险就会降低。