Magento 2に統合されたモジュールにカスタムXMLファイルを含めるにはどうすればよいですか?(MageStackDayミステリー質問2)


22

500ptsバウンティのMageStackDayボーナス質問と、1年間無料のZ-Rayライセンスを獲得する可能性。詳細はこちら >>をご覧ください <<

質問は、Magento 2のコア開発者であるAnton Krilによって提供されました。

質問:

個別の構成セットを持つ拡張機能を作成しています。
この手段は、私が使用することはできませんconfig.xmlか、routes.xmlまたはfieldset.xmlまたは持つMagentoの他の設定xmlファイル。
例。

行と列を持つ「テーブル」設定を定義しているとしましょう。このxmlを以下で使用できます。(それを呼び出すtable.xml

<table xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="path/to/table.xsd">
    <row id="row1">
        <column id="col1" sort="10" attr1="val1">
            <label>Col 1</label>
        </column>
    </row>
    <row id="row2">
        <column id="col1" sort="10" attr1="val1">
            <label>Col 1</label>
        </column>
        <column id="col2" sort="20" disabled="true" attr1="val2" >
            <label>Col 2</label>
        </column>
        <column id="col3" sort="15" attr1="val1">
            <label>Col 3</label>
        </column>
    </row>
</table>

しかし、他の拡張機能が含まれている場合、table.xml構成リーダーでそれを取得し、2つ以上のxmlファイルをマージする必要があります。2番目のファイルがこのように見える場合

<table xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="path/to/table.xsd">
    <row id="row1">
        <column id="col2" sort="10" attr1="val2">
            <label>Col 2</label>
        </column>
    </row>
    <row id="row2">
        <column id="col1" sort="10" attr1="val5" />
    </row>
</table>

その結果、2番目の列が最初の行に追加され、値attr1が2番目のxmlによって上書きされます。

<table ....>
    <row id="row1">
        <column id="col1" sort="10" attr1="val1"> <!-- from first xml -->
            <label>Col 1</label>
        </column>
        <column id="col2" sort="10" attr1="val2"><!-- from second xml-->
            <label>Col 2</label>
        </column>
    </row>
    <row id="row2">
        <column id="col1" sort="10" attr1="val5"><!--they apear in both xmls with the same path and id and second one overrides the value for `attr1`-->
            <label>Col 1</label>
        </column>
        <column id="col2" sort="20" disabled="true" attr1="val2"><!-- from first xml -->
            <label>Col 2</label>
        </column>
        <column id="col3" sort="15" attr1="val1"><!-- from first xml -->
            <label>Col 3</label>
        </column>
    </row>
</table>

Magento 1では、電話するだけでこれを行うことができました。

 $merged = Mage::getConfig()->loadModulesConfiguration('table.xml')
            ->applyExtends();

Magento 2でも同じことができますか?

回答:


15

Magento 2では、これは\Magento\Framework\Config\Reader\Filesystemクラスによって処理されます。このクラスを使用すると、マージするxmlファイルを指定できます。

次の部分は、使用可能なモジュールで見つかったすべてのファイルをマージし、出力(からのスニペット\Magento\Framework\Config\Reader\Filesystem)をマージします

/**
 * Load configuration scope
 *
 * @param string|null $scope
 * @return array
 */
public function read($scope = null)
{
    $scope = $scope ?: $this->_defaultScope;
    $fileList = $this->_fileResolver->get($this->_fileName, $scope);
    if (!count($fileList)) {
        return [];
    }
    $output = $this->_readFiles($fileList);

    return $output;
}

/**
 * Read configuration files
 *
 * @param array $fileList
 * @return array
 * @throws \Magento\Framework\Exception
 */
protected function _readFiles($fileList)
{
    /** @var \Magento\Framework\Config\Dom $configMerger */
    $configMerger = null;
    foreach ($fileList as $key => $content) {
        try {
            if (!$configMerger) {
                $configMerger = $this->_createConfigMerger($this->_domDocumentClass, $content);
            } else {
                $configMerger->merge($content);
            }
        } catch (\Magento\Framework\Config\Dom\ValidationException $e) {
            throw new \Magento\Framework\Exception("Invalid XML in file " . $key . ":\n" . $e->getMessage());
        }
    }
    if ($this->_isValidated) {
        $errors = [];
        if ($configMerger && !$configMerger->validate($this->_schemaFile, $errors)) {
            $message = "Invalid Document \n";
            throw new \Magento\Framework\Exception($message . implode("\n", $errors));
        }
    }

    $output = [];
    if ($configMerger) {
        $output = $this->_converter->convert($configMerger->getDom());
    }
    return $output;
}

私が作成したソリューションでは、上記のクラスを拡張して、必要なxmlファイルを提供し、検証するxsdファイルの場所を指定します(完全な例については、https://github.com/Genmato/MageStackTableを参照してください)。

namespace Genmato\TableXml\Model\Table;

class Reader extends \Magento\Framework\Config\Reader\Filesystem
{
    protected $_idAttributes = [
        '/table/row' => 'id',
        '/table/row/column' => 'id',
    ];

    /**
     * @param \Magento\Framework\Config\FileResolverInterface $fileResolver
     * @param \Magento\Framework\Config\ConverterInterface $converter
     * @param \Genmato\TableXml\Model\Table\SchemaLocator $schemaLocator
     * @param \Magento\Framework\Config\ValidationStateInterface $validationState
     * @param string $fileName
     * @param array $idAttributes
     * @param string $domDocumentClass
     * @param string $defaultScope
     */
    public function __construct(
        \Magento\Framework\Config\FileResolverInterface $fileResolver,
        \Magento\Framework\Config\ConverterInterface $converter,
        \Genmato\TableXml\Model\Table\SchemaLocator $schemaLocator,
        \Magento\Framework\Config\ValidationStateInterface $validationState,
        $fileName = 'table.xml',
        $idAttributes = [],
        $domDocumentClass = 'Magento\Framework\Config\Dom',
        $defaultScope = 'global'
    ) {
        parent::__construct(
            $fileResolver,
            $converter,
            $schemaLocator,
            $validationState,
            $fileName,
            $idAttributes,
            $domDocumentClass,
            $defaultScope
        );
    }

マージされたデータを取得するには、次を呼び出します。

$output = $this->_objectManager->get('Genmato\TableXml\Model\Table\Reader')->read();

出力は、マージされたxmlの配列表現です。

編集:

ファイルの読み取り方法をテストするために、実際の例を作成しました(https://github.com/Genmato/MageStackTableを参照)。ソリューションのビルドに関する回答を更新しました。


ウラジミール、今日はDomクラスの例であなたの以前の回答バージョンを見ました。私はReaderクラスを活用した回答に取り組み始めました。その間、私は質問ページを更新し、あなたがそれをやったことに気付きました:-) +1
Wojtek Naruniec

詳細な回答とgithubのPOCモジュールに感謝します。将来の参考のためにそこに残してください。ここに…賞金があります。
マリウス

マリウス、ありがとう!モジュールはGitHubで利用可能のままにします。
ウラジミールケルホフ
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.