diff --git a/config/vue3migration/compiler.js b/config/vue3migration/compiler.js
new file mode 100644
index 0000000000000000000000000000000000000000..bb92e1e2356a8ef637dc6447c5a5cd4563212d7c
--- /dev/null
+++ b/config/vue3migration/compiler.js
@@ -0,0 +1,50 @@
+const { parse, compile: compilerDomCompile } = require('@vue/compiler-dom');
+
+const getPropIndex = (node, prop) => node.props?.findIndex((p) => p.name === prop) ?? -1;
+
+function modifyKeysInsideTemplateTag(templateNode) {
+  let keyCandidate = null;
+  for (const node of templateNode.children) {
+    const keyBindingIndex = node.props
+      ? node.props.findIndex((prop) => prop.arg && prop.arg.content === 'key')
+      : -1;
+
+    if (keyBindingIndex !== -1 && getPropIndex(node, 'for') === -1) {
+      if (!keyCandidate) {
+        keyCandidate = node.props[keyBindingIndex];
+      }
+      node.props.splice(keyBindingIndex, 1);
+    }
+  }
+
+  if (keyCandidate) {
+    templateNode.props.push(keyCandidate);
+  }
+}
+
+module.exports = {
+  parse,
+  compile(template, options) {
+    const rootNode = parse(template, options);
+    const pendingNodes = [rootNode];
+    while (pendingNodes.length) {
+      const currentNode = pendingNodes.pop();
+      if (getPropIndex(currentNode, 'for') !== -1) {
+        if (currentNode.tag === 'template') {
+          // This one will be dropped all together with compiler when we drop Vue.js 2 support
+          modifyKeysInsideTemplateTag(currentNode);
+        }
+
+        // This one will be dropped when https://github.com/vuejs/core/issues/7725 will be fixed
+        const vOncePropIndex = getPropIndex(currentNode, 'once');
+        if (vOncePropIndex !== -1) {
+          currentNode.props.splice(vOncePropIndex, 1);
+        }
+      }
+
+      currentNode.children?.forEach((child) => pendingNodes.push(child));
+    }
+
+    return compilerDomCompile(rootNode, options);
+  },
+};
diff --git a/config/webpack.config.js b/config/webpack.config.js
index e60440e74b3f16f4d424bfdff8f95f59d4c46463..53479a5a9ba57f63ac8f0f6dffb775c8d61447b1 100644
--- a/config/webpack.config.js
+++ b/config/webpack.config.js
@@ -301,12 +301,15 @@ if (WEBPACK_USE_ESBUILD_LOADER) {
 }
 
 const vueLoaderOptions = {
+  ident: 'vue-loader-options',
+
   cacheDirectory: path.join(CACHE_PATH, 'vue-loader'),
   cacheIdentifier: [
     process.env.NODE_ENV || 'development',
     webpack.version,
     VUE_VERSION,
     VUE_LOADER_VERSION,
+    EXPLICIT_VUE_VERSION,
   ].join('|'),
 };
 
@@ -334,6 +337,8 @@ if (USE_VUE3) {
   Object.assign(alias, {
     vue: '@vue/compat',
   });
+
+  vueLoaderOptions.compiler = require.resolve('./vue3migration/compiler');
 }
 
 module.exports = {
diff --git a/jest.config.base.js b/jest.config.base.js
index 611853c019cc3e1a4a4a6ae897fbc008f83d573d..50fd91b2e38e4291df740d905a5a532decc8aaac 100644
--- a/jest.config.base.js
+++ b/jest.config.base.js
@@ -44,6 +44,7 @@ module.exports = (path, options = {}) => {
     Object.assign(globals, {
       'vue-jest': {
         experimentalCSSCompile: false,
+        compiler: require.resolve('./config/vue3migration/compiler'),
         compilerOptions: {
           compatConfig: {
             MODE: 2,
diff --git a/patches/@vue+vue3-jest+29.2.3.patch b/patches/@vue+vue3-jest+29.2.3.patch
new file mode 100644
index 0000000000000000000000000000000000000000..3b0ced984000d858c9b23ca16efaaa5ebd8cc833
--- /dev/null
+++ b/patches/@vue+vue3-jest+29.2.3.patch
@@ -0,0 +1,22 @@
+diff --git a/node_modules/@vue/vue3-jest/lib/process.js b/node_modules/@vue/vue3-jest/lib/process.js
+index a8d1c5c..a6b2036 100644
+--- a/node_modules/@vue/vue3-jest/lib/process.js
++++ b/node_modules/@vue/vue3-jest/lib/process.js
+@@ -108,12 +108,17 @@ function processTemplate(descriptor, filename, config) {
+     (descriptor.script && descriptor.script.lang)
+   const isTS = /^typescript$|tsx?$/.test(lang)
+ 
++  const compiler = typeof vueJestConfig.compiler === 'string'
++    ? require(vueJestConfig.compiler)
++    : vueJestConfig.compiler
++
+   const result = compileTemplate({
+     id: filename,
+     source: template.content,
+     filename,
+     preprocessLang: template.lang,
+     preprocessOptions: vueJestConfig[template.lang],
++    compiler,
+     compilerOptions: {
+       bindingMetadata: bindings,
+       mode: 'module',