diff --git a/core/composer.json b/core/composer.json
index c3600ab..09b4b9e 100644
--- a/core/composer.json
+++ b/core/composer.json
@@ -120,6 +120,7 @@
         "drupal/migrate_drupal": "self.version",
         "drupal/migrate_drupal_ui": "self.version",
         "drupal/node": "self.version",
+        "drupal/node_translation_redirect": "self.version",
         "drupal/options": "self.version",
         "drupal/outside_in": "self.version",
         "drupal/page_cache": "self.version",
diff --git a/core/modules/node/node.services.yml b/core/modules/node/node.services.yml
index 2ff32c3..a402537 100644
--- a/core/modules/node/node.services.yml
+++ b/core/modules/node/node.services.yml
@@ -50,3 +50,8 @@ services:
     arguments: ['@current_route_match']
     tags:
       - { name: 'context_provider' }
+  node.node_translation_redirect_subscriber:
+    class: Drupal\node\EventSubscriber\NodeTranslationRedirectSubscriber
+    arguments: ["@keyvalue", "@module_handler", "@module_installer"]
+    tags:
+      - { name: event_subscriber }
diff --git a/core/modules/node/src/EventSubscriber/NodeTranslationRedirectSubscriber.php b/core/modules/node/src/EventSubscriber/NodeTranslationRedirectSubscriber.php
new file mode 100644
index 0000000..7f37883
--- /dev/null
+++ b/core/modules/node/src/EventSubscriber/NodeTranslationRedirectSubscriber.php
@@ -0,0 +1,100 @@
+<?php
+
+namespace Drupal\node\EventSubscriber;
+
+use Drupal\Core\Extension\ModuleHandlerInterface;
+use Drupal\Core\Extension\ModuleInstallerInterface;
+use Drupal\Core\KeyValueStore\KeyValueFactoryInterface;
+use Drupal\migrate\Event\MigrateEvents;
+use Drupal\migrate\Event\MigrateImportEvent;
+use Drupal\migrate\Event\MigratePostRowSaveEvent;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+
+/**
+ * Creates a key value collection for migrated node translation redirections.
+ */
+class NodeTranslationRedirectSubscriber implements EventSubscriberInterface {
+
+  /**
+   * The key value factory.
+   *
+   * @var \Drupal\Core\KeyValueStore\KeyValueFactoryInterface
+   */
+  protected $keyValue;
+
+  /**
+   * The module handler.
+   *
+   * @var \Drupal\Core\Extension\ModuleHandlerInterface
+   */
+  protected $moduleHandler;
+
+  /**
+   * The module installer.
+   *
+   * @var \Drupal\Core\Extension\ModuleInstallerInterface
+   */
+  protected $moduleInstaller;
+
+  /**
+   * Construct a NodeTranslationSubscriber.
+   *
+   * @param \Drupal\Core\KeyValueStore\KeyValueFactoryInterface $key_value
+   *   The key value factory.
+   * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler
+   *   The module handler.
+   * @param \Drupal\Core\Extension\ModuleInstallerInterface $module_installer
+   *   The module installer.
+   */
+  public function __construct(KeyValueFactoryInterface $key_value, ModuleHandlerInterface $module_handler, ModuleInstallerInterface $module_installer) {
+    $this->keyValue = $key_value;
+    $this->moduleHandler = $module_handler;
+    $this->moduleInstaller = $module_installer;
+  }
+
+  /**
+   * Maps the old nid to the new one in the key value collection.
+   *
+   * @param \Drupal\migrate\Event\MigratePostRowSaveEvent $event
+   *   The migrate post row save event.
+   */
+  public function onPostRowSave(MigratePostRowSaveEvent $event) {
+    $configuration = $event->getMigration()->getDestinationConfiguration();
+    if ($configuration['plugin'] == 'entity:node' && !empty($configuration['translations'])) {
+      $row = $event->getRow();
+      $source = $row->getSource();
+      $destination = $row->getDestination();
+      $collection = $this->keyValue->get('node_translation_redirect');
+      $collection->set($source['nid'], [$destination['nid'], $destination['langcode']]);
+    }
+  }
+
+  /**
+   * Enables the node_translation_redirect module to handle the redirections.
+   *
+   * @param \Drupal\migrate\Event\MigrateImportEvent $event
+   *   The migrate import event.
+   */
+  public function onPostImport(MigrateImportEvent $event) {
+    if (substr($event->getMigration()->getPluginId(), 2, 17) == '_node_translation') {
+      if (!$this->moduleHandler->moduleExists('node_translation_redirect')) {
+        $this->moduleInstaller->install(['node_translation_redirect']);
+      }
+    }
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getSubscribedEvents() {
+    $events = [];
+
+    if (class_exists('\Drupal\migrate\Event\MigrateEvents')) {
+      $events[MigrateEvents::POST_ROW_SAVE] = ['onPostRowSave'];
+      $events[MigrateEvents::POST_IMPORT] = ['onPostImport'];
+    }
+
+    return $events;
+  }
+
+}
diff --git a/core/modules/node/tests/src/Kernel/Migrate/d6/MigrateNodeTest.php b/core/modules/node/tests/src/Kernel/Migrate/d6/MigrateNodeTest.php
index 590064d..788be5a 100644
--- a/core/modules/node/tests/src/Kernel/Migrate/d6/MigrateNodeTest.php
+++ b/core/modules/node/tests/src/Kernel/Migrate/d6/MigrateNodeTest.php
@@ -30,6 +30,8 @@ protected function setUp() {
     $this->installSchema('file', ['file_usage']);
     $this->executeMigrations([
       'language',
+      'd6_language_types',
+      'd6_language_negotiation_settings',
       'd6_language_content_settings',
       'd6_node',
       'd6_node_translation',
diff --git a/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeTest.php b/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeTest.php
index 9ca1d4a..a74f2d4 100644
--- a/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeTest.php
+++ b/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeTest.php
@@ -47,6 +47,8 @@ protected function setUp() {
 
     $this->executeMigrations([
       'language',
+      'd7_language_types',
+      'd7_language_negotiation_settings',
       'd7_user_role',
       'd7_user',
       'd7_node_type',
diff --git a/core/modules/node_translation_redirect/node_translation_redirect.info.yml b/core/modules/node_translation_redirect/node_translation_redirect.info.yml
new file mode 100644
index 0000000..67c518f
--- /dev/null
+++ b/core/modules/node_translation_redirect/node_translation_redirect.info.yml
@@ -0,0 +1,7 @@
+name: Node translation redirect
+type: module
+description: 'Redirects non-existent migrated node translations.'
+package: Core
+version: VERSION
+core: 8.x
+hidden: true
diff --git a/core/modules/node_translation_redirect/node_translation_redirect.services.yml b/core/modules/node_translation_redirect/node_translation_redirect.services.yml
new file mode 100644
index 0000000..985ea23
--- /dev/null
+++ b/core/modules/node_translation_redirect/node_translation_redirect.services.yml
@@ -0,0 +1,7 @@
+services:
+  node_translation_redirect.exception_subscriber:
+    class: Drupal\node_translation_redirect\EventSubscriber\ExceptionSubscriber
+    arguments: ["@keyvalue", "@language_manager", "@url_generator"]
+    tags:
+      - { name: event_subscriber }
+
diff --git a/core/modules/node_translation_redirect/src/EventSubscriber/ExceptionSubscriber.php b/core/modules/node_translation_redirect/src/EventSubscriber/ExceptionSubscriber.php
new file mode 100644
index 0000000..db1d7d9
--- /dev/null
+++ b/core/modules/node_translation_redirect/src/EventSubscriber/ExceptionSubscriber.php
@@ -0,0 +1,89 @@
+<?php
+
+namespace Drupal\node_translation_redirect\EventSubscriber;
+
+use Drupal\Core\KeyValueStore\KeyValueFactoryInterface;
+use Drupal\Core\Routing\UrlGeneratorInterface;
+use Drupal\language\ConfigurableLanguageManagerInterface;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+use Symfony\Component\HttpFoundation\RedirectResponse;
+use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
+use Symfony\Component\HttpKernel\KernelEvents;
+
+/**
+ * Redirects non-existent migrated node translations.
+ */
+class ExceptionSubscriber implements EventSubscriberInterface {
+
+  /**
+   * The key value factory.
+   *
+   * @var \Drupal\Core\KeyValueStore\KeyValueFactoryInterface
+   */
+  protected $keyValue;
+
+  /**
+   * The language manager.
+   *
+   * @var \Drupal\language\ConfigurableLanguageManagerInterface
+   */
+  protected $languageManager;
+
+  /**
+   * The URL generator.
+   *
+   * @var \Drupal\Core\Routing\UrlGeneratorInterface
+   */
+  protected $urlGenerator;
+
+  /**
+   * Construct an ExceptionSubscriber.
+   *
+   * @param \Drupal\Core\KeyValueStore\KeyValueFactoryInterface $key_value
+   *   The key value factory.
+   * @param \Drupal\language\ConfigurableLanguageManagerInterface $language_manager
+   *   The language manager.
+   * @param \Drupal\Core\Routing\UrlGeneratorInterface $url_generator
+   *   The URL generator.
+   */
+  public function __construct(KeyValueFactoryInterface $key_value, ConfigurableLanguageManagerInterface $language_manager, UrlGeneratorInterface $url_generator) {
+    $this->keyValue = $key_value;
+    $this->languageManager = $language_manager;
+    $this->urlGenerator = $url_generator;
+  }
+
+  /**
+   * Redirects not found node translations using the key value collection.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent $event;
+   *   The exception event.
+   */
+  public function onException(GetResponseForExceptionEvent $event) {
+    $status_code = $event->getException()->getStatusCode();
+    $path = $event->getRequest()->getPathInfo();
+
+    // Path aliases have already been migrated, so the path we're getting from
+    // getPathInfo() is the internal path.
+    if ($status_code == 404 && preg_match('/^\/node\/(\d+)$/', $path, $matches)) {
+      $collection = $this->keyValue->get('node_translation_redirect');
+      $old_nid = isset($matches[1]) ? $matches[1] : NULL;
+      if ($old_nid && $collection->has($old_nid)) {
+        list($nid, $langcode) = $collection->get($old_nid);
+        $language = $this->languageManager->getLanguage($langcode);
+        $url = $this->urlGenerator->generateFromRoute('entity.node.canonical', ['node' => $nid], ['language' => $language]);
+        $response = new RedirectResponse($url);
+        $response->send();
+      }
+    }
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getSubscribedEvents() {
+    $events[KernelEvents::EXCEPTION] = ['onException'];
+
+    return $events;
+  }
+
+}
diff --git a/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d6/NodeTranslationRedirectTest.php b/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d6/NodeTranslationRedirectTest.php
new file mode 100644
index 0000000..3b80bad
--- /dev/null
+++ b/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d6/NodeTranslationRedirectTest.php
@@ -0,0 +1,68 @@
+<?php
+
+namespace Drupal\Tests\node_translation_redirect\Kernel\Migrate\d6;
+
+use Drupal\Tests\migrate_drupal\Kernel\d6\MigrateDrupal6TestBase;
+use Symfony\Component\HttpFoundation\Request;
+
+/**
+ * Tests node translation redirections.
+ *
+ * @group node_translation_redirect
+ */
+class NodeTranslationRedirectTest extends MigrateDrupal6TestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static $modules = [
+    'content_translation',
+    'language',
+    'menu_ui',
+  ];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    parent::setUp();
+
+    $this->installEntitySchema('node');
+    $this->installConfig(['node']);
+    $this->installSchema('node', ['node_access']);
+    $this->installSchema('system', ['key_value']);
+    $this->migrateUsers(FALSE);
+    $this->migrateFields();
+
+    $this->executeMigrations([
+      'language',
+      'd6_language_types',
+      'd6_language_negotiation_settings',
+      'd6_node_settings',
+      'd6_node',
+      'd6_node_translation',
+    ]);
+  }
+
+  /**
+   * Tests that not found node translations are redirected.
+   */
+  public function testNodeTranslationRedirect() {
+    $request = Request::create('/node/11');
+    $kernel = $this->container->get('http_kernel');
+    $this->expectOutputString('<!DOCTYPE html>
+<html>
+    <head>
+        <meta charset="UTF-8" />
+        <meta http-equiv="refresh" content="1;url=/fr/node/10" />
+
+        <title>Redirecting to /fr/node/10</title>
+    </head>
+    <body>
+        Redirecting to <a href="/fr/node/10">/fr/node/10</a>.
+    </body>
+</html>');
+    $response = $kernel->handle($request);
+  }
+
+}
diff --git a/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d7/NodeTranslationRedirectTest.php b/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d7/NodeTranslationRedirectTest.php
new file mode 100644
index 0000000..3748c1b
--- /dev/null
+++ b/core/modules/node_translation_redirect/tests/src/Kernel/Migrate/d7/NodeTranslationRedirectTest.php
@@ -0,0 +1,70 @@
+<?php
+
+namespace Drupal\Tests\node_translation_redirect\Kernel\Migrate\d7;
+
+use Drupal\Tests\migrate_drupal\Kernel\d7\MigrateDrupal7TestBase;
+use Symfony\Component\HttpFoundation\Request;
+
+/**
+ * Tests node translation redirections.
+ *
+ * @group node_translation_redirect
+ */
+class NodeTranslationRedirectTest extends MigrateDrupal7TestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static $modules = [
+    'content_translation',
+    'language',
+    'menu_ui',
+    'node',
+    'text',
+  ];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    parent::setUp();
+
+    $this->installEntitySchema('node');
+    $this->installConfig('node');
+    $this->installSchema('node', ['node_access']);
+    $this->installSchema('system', ['key_value']);
+
+    $this->executeMigrations([
+      'language',
+      'd7_language_types',
+      'd7_language_negotiation_settings',
+      'd7_user_role',
+      'd7_user',
+      'd7_node_type',
+      'd7_node',
+      'd7_node_translation',
+    ]);
+  }
+
+  /**
+   * Tests that not found node translations are redirected.
+   */
+  public function testNodeTranslationRedirect() {
+    $request = Request::create('/node/3');
+    $kernel = $this->container->get('http_kernel');
+    $this->expectOutputString('<!DOCTYPE html>
+<html>
+    <head>
+        <meta charset="UTF-8" />
+        <meta http-equiv="refresh" content="1;url=/is/node/2" />
+
+        <title>Redirecting to /is/node/2</title>
+    </head>
+    <body>
+        Redirecting to <a href="/is/node/2">/is/node/2</a>.
+    </body>
+</html>');
+    $response = $kernel->handle($request);
+  }
+
+}
diff --git a/core/modules/path/tests/src/Kernel/Migrate/d6/MigrateUrlAliasTest.php b/core/modules/path/tests/src/Kernel/Migrate/d6/MigrateUrlAliasTest.php
index 092d5ac..95d0d80 100644
--- a/core/modules/path/tests/src/Kernel/Migrate/d6/MigrateUrlAliasTest.php
+++ b/core/modules/path/tests/src/Kernel/Migrate/d6/MigrateUrlAliasTest.php
@@ -31,6 +31,8 @@ protected function setUp() {
 
     $this->executeMigrations([
       'language',
+      'd6_language_types',
+      'd6_language_negotiation_settings',
       'd6_node_settings',
       'd6_node',
       'd6_node_translation',
diff --git a/core/modules/path/tests/src/Kernel/Migrate/d7/MigrateUrlAliasTest.php b/core/modules/path/tests/src/Kernel/Migrate/d7/MigrateUrlAliasTest.php
index 2edf258..d8719b0 100644
--- a/core/modules/path/tests/src/Kernel/Migrate/d7/MigrateUrlAliasTest.php
+++ b/core/modules/path/tests/src/Kernel/Migrate/d7/MigrateUrlAliasTest.php
@@ -35,6 +35,8 @@ protected function setUp() {
 
     $this->executeMigrations([
       'language',
+      'd7_language_types',
+      'd7_language_negotiation_settings',
       'd7_user_role',
       'd7_user',
       'd7_node_type',
