From 0dcfdbd5372dfd0c9f414a77c45f26295724416c Mon Sep 17 00:00:00 2001 From: Benjamin Rasmussen Date: Tue, 17 Dec 2024 14:31:54 +0100 Subject: [PATCH] Update BnfImporter to also import text paragraphs. This feels a bit dirty, but for now, until we've cleared how to deal with GraphQL schemas/codegen, it'll do. Only text_body paragraphs are imported. --- .../bnf_client/src/Services/BnfExporter.php | 1 + .../custom/bnf/src/Services/BnfImporter.php | 144 ++++++++++++++++-- 2 files changed, 133 insertions(+), 12 deletions(-) diff --git a/web/modules/custom/bnf/bnf_client/src/Services/BnfExporter.php b/web/modules/custom/bnf/bnf_client/src/Services/BnfExporter.php index b4803e9f7..6886b86d6 100644 --- a/web/modules/custom/bnf/bnf_client/src/Services/BnfExporter.php +++ b/web/modules/custom/bnf/bnf_client/src/Services/BnfExporter.php @@ -56,6 +56,7 @@ public function exportNode(NodeInterface $node): void { try { $bnfServer = (string) getenv('BNF_SERVER_GRAPHQL_ENDPOINT'); + $bnfServer = 'https://dapple-cms.local/graphql'; if (!filter_var($bnfServer, FILTER_VALIDATE_URL)) { throw new \InvalidArgumentException('The provided BNF server URL is not valid.'); diff --git a/web/modules/custom/bnf/src/Services/BnfImporter.php b/web/modules/custom/bnf/src/Services/BnfImporter.php index 138db8519..371556350 100644 --- a/web/modules/custom/bnf/src/Services/BnfImporter.php +++ b/web/modules/custom/bnf/src/Services/BnfImporter.php @@ -6,10 +6,12 @@ use Drupal\bnf\Exception\AlreadyExistsException; use Drupal\Core\Entity\EntityTypeManagerInterface; use Drupal\Core\StringTranslation\TranslationInterface; +use Drupal\paragraphs\ParagraphInterface; use GuzzleHttp\ClientInterface; use Psr\Log\LoggerInterface; use function Safe\json_decode; use function Safe\parse_url; +use function Safe\preg_replace; /** * Service related to importing content from an external source. @@ -20,6 +22,15 @@ */ class BnfImporter { + const ALLOWED_PARAGRAPHS = [ + 'text_body', + ]; + + /** + * The BNF UUID of the content that we import. + */ + protected string $uuid; + /** * Constructor. */ @@ -30,10 +41,128 @@ public function __construct( protected LoggerInterface $logger, ) {} + /** + * Building the query we use to get data from source. + */ + protected function getQuery(string $queryName): string { + // Example of GraphQL query: "nodeArticle". + return <<uuid") { + title + paragraphs { + ... on ParagraphTextBody { + __typename + body { + format, + value + } + } + } + } + } + GRAPHQL; + + } + + /** + * Parses paragraphs from GraphQL node data into Drupal-compatible structures. + * + * @param mixed[] $nodeData + * The GraphQL node data containing paragraphs. + * + * @return mixed[] + * Array of paragraph values, that we can use to create paragraph entities. + */ + protected function parseParagraphs(array $nodeData) { + $parsedParagraphs = []; + + // Ensure paragraphs exist in the GraphQL response. + if (empty($nodeData['paragraphs'])) { + return $parsedParagraphs; + } + + foreach ($nodeData['paragraphs'] as $paragraphData) { + $type = $paragraphData['__typename'] ?? ''; + + // Convert typename to Drupal paragraph bundle name. + $bundleName = $this->graphqlTypeToBundle($type); + + if (!in_array($bundleName, self::ALLOWED_PARAGRAPHS)) { + continue; + } + + $paragraph = ['type' => $bundleName]; + + // Map fields dynamically. + foreach ($paragraphData as $key => $value) { + if ($key === '__typename') { + continue; + } + + // Assume Drupal uses field names like "field_{key}". + $drupalFieldName = 'field_' . $key; + $paragraph[$drupalFieldName] = $value; + } + + $parsedParagraphs[] = $paragraph; + } + + return $parsedParagraphs; + } + + /** + * Creating the paragraphs, that we will add to the nodes. + * + * @param mixed[] $nodeData + * The GraphQL node data containing paragraphs. + * + * @return \Drupal\paragraphs\ParagraphInterface[] + * The paragraph entities. + */ + protected function getParagraphs(array $nodeData): array { + $parsedParagraphs = $this->parseParagraphs($nodeData); + $storage = $this->entityTypeManager->getStorage('paragraph'); + $paragraphs = []; + foreach ($parsedParagraphs as $paragraphData) { + $paragraph = $storage->create($paragraphData); + + if ($paragraph instanceof ParagraphInterface) { + $paragraph->save(); + $paragraphs[] = $paragraph; + } + } + + return $paragraphs; + } + + /** + * Converts a GraphQL typename to a Drupal paragraph bundle name. + * + * @param string $typeName + * The GraphQL typename (e.g., ParagraphTextBody). + * + * @return string + * The Drupal paragraph bundle name (e.g., text_body). + */ + protected function graphqlTypeToBundle(string $typeName): string { + // Removing 'Paragraph' prefix. + $typeName = preg_replace('/^Paragraph/', '', $typeName); + + // Converting CamelCase to snake_case. + $pattern = '/(?<=\\w)(?=[A-Z])|(?<=[a-z])(?=[0-9])/'; + $typeName = preg_replace($pattern, '_', $typeName); + + return strtolower($typeName); + } + /** * Importing a node from a GraphQL source endpoint. */ public function importNode(string $uuid, string $endpointUrl, string $nodeType = 'article'): void { + $this->uuid = $uuid; + $queryName = 'node' . ucfirst($nodeType); + $nodeStorage = $this->entityTypeManager->getStorage('node'); $existingNodes = @@ -48,18 +177,6 @@ public function importNode(string $uuid, string $endpointUrl, string $nodeType = throw new AlreadyExistsException('Cannot import node - already exists.'); } - // Example of GraphQL query: "nodeArticle". - $queryName = 'node' . ucfirst($nodeType); - - // For now, we only support the title of the nodes. - $query = <<getQuery($queryName); + $response = $this->httpClient->request('post', $endpointUrl, [ 'headers' => [ 'Content-Type' => 'application/json', @@ -95,6 +214,7 @@ public function importNode(string $uuid, string $endpointUrl, string $nodeType = try { $nodeData['type'] = $nodeType; $nodeData['uuid'] = $uuid; + $nodeData['field_paragraphs'] = $this->getParagraphs($nodeData); $node = $nodeStorage->create($nodeData); $node->save();