sistema para operativo nodejs node modulos framework documentación docs diccionario actualizar node.js express reactjs webpack isomorphic-javascript

node.js - para - Advertencia: Reacción intentó reutilizar marcado en un contenedor pero la suma de comprobación no era válida



nodejs manual pdf (7)

En mi caso, el problema fue causado por el hecho de que estaba usando el componente MediaQuery de ''reaccionar-receptivo'' sin pasar la propiedad ''valor'' que es usado por el componente cuando no puede acceder al ancho de la pantalla (por ejemplo, en el Servidor )

Estoy intentando hacer funcionar una aplicación isomorfa Node.js, Express, Webpack, React. Me aparece el siguiente error. ¿Alguna sugerencia sobre cómo solucionarlo?

Warning: React attempted to reuse markup in a container but the checksum was invalid. This generally means that you are using server rendering and the markup generated on the server was not what the client was expecting. React injected new markup to compensate which works but you have lost many of the benefits of server rendering. Instead, figure out why the markup being generated is different on the client or server: (client) rgin:0;display:flex;-webkit-align-items: (server) rgin:0;display:flex;align-items:center;j warning @ warning.js:45 ReactMount._mountImageIntoNode @ ReactMount.js:807 wrapper @ ReactPerf.js:66 mountComponentIntoNode @ ReactMount.js:268 Mixin.perform @ Transaction.js:136 batchedMountComponentIntoNode @ ReactMount.js:282 Mixin.perform @ Transaction.js:136 ReactDefaultBatchingStrategy.batchedUpdates @ ReactDefaultBatchingStrategy.js:62 batchedUpdates @ ReactUpdates.js:94 ReactMount._renderNewRootComponent @ ReactMount.js:476 wrapper @ ReactPerf.js:66 ReactMount._renderSubtreeIntoContainer @ ReactMount.js:550 ReactMount.render @ ReactMount.js:570 wrapper @ ReactPerf.js:66 (anonymous function) @ client.jsx:14 (anonymous function) @ iso.js:120 each @ iso.js:21 bootstrap @ iso.js:111 (anonymous function) @ client.jsx:12 __webpack_require__ @ bootstrap d56606d95d659f2e05dc:19 (anonymous function) @ bootstrap d56606d95d659f2e05dc:39 (anonymous function) @ bootstrap d56606d95d659f2e05dc:39

Esto es lo que el servidor está entregando inicialmente al navegador:

<!doctype html> <html lang=""> <head> <title>my title</title> <meta name="apple-mobile-web-app-title" content="my title" data-react-helmet="true" /> <meta name="apple-mobile-web-app-status-bar-style" content="black" data-react-helmet="true" /> <meta name="apple-mobile-web-app-capable" content="yes" data-react-helmet="true" /> <meta name="mobile-web-app-capable" content="yes" data-react-helmet="true" /> <meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no" data-react-helmet="true" /> <meta name="description" content="my description." data-react-helmet="true" /> <meta http-equiv="X-UA-Compatible" content="IE=edge" data-react-helmet="true" /> <meta charset="utf-8" data-react-helmet="true" /> <link rel="stylesheet" href="/assets/styles/reset.css" data-react-helmet="true" /> <link rel="stylesheet" href="/assets/styles/base.css" data-react-helmet="true" /> <link rel="stylesheet" href="/assets/styles/Carousel.css" data-react-helmet="true" /> <link rel="stylesheet" href="/assets/styles/main.css" data-react-helmet="true" /> <link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto+Condensed" type="text/css" data-react-helmet="true" /> <link rel="icon" href="/assets/185bb6f691241307862b331970a6bff1.ico" type="image/x-icon" data-react-helmet="true" /> SCRIPT </head> <body> <script src="https://cdn.firebase.com/js/client/2.2.7/firebase.js"></script> <script src="https://cdn.firebase.com/libs/reactfire/0.4.0/reactfire.min.js"></script> <div class="app"> <div class="___iso-html___" data-key="_0"><div data-reactid=".1hkqsbm9n9c" data-react-checksum="794698749"><div data-reactid=".1hkqsbm9n9c.0"><div data-reactid=".1hkqsbm9n9c.0.$=10"></div><div style="position:fixed;z-index:2;top:0;left:0;right:0;height:60px;color:rgb(219,219,219);font-family:mainnextcondensed_ultralight;font-size:17px;overflow:hidden;" data-reactid=".1hkqsbm9n9c.0.$/=11"><div style="position:absolute;left:0;top:0;background-color:rgba(27,27,27,0.92);padding-right:35px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=10"><div style="float:left;height:60px;width:13px;border-left:5px solid rgb(210,45,164);" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=10.$/=10"></div><div style="float:left;height:60px;width:227px;background-image:url();background-repeat:no-repeat;background-position:center;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=10.$/=11"></div><div style="display:none;width:0;height:0;border-style:solid;border-width:6px 6px 0 6px;border-color:rgb(117,117,117) transparent transparent transparent;-webkit-transform:rotate(360deg);float:left;margin-left:6px;margin-top:26px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=10.$/=12"></div></div><div style="position:absolute;top:0px;left:280px;width:340px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11"><div style="background-color:rgba(27,27,27,0.92);height:10px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=10"></div><div style="background-color:rgba(53,53,53,0.84);height:40px;position:relative;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=11"><div style="position:absolute;top:0;bottom:0;left:0;right:0;padding:0;margin:0;display:flex;align-items:center;justify-content:center;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=11.$=10"><div style="background-image:url(&#x27;/assets/3bec3e57cb5ee05658440d21984fb7b7.png&#x27;);background-repeat:no-repeat;background-position:-58px -194px;width:23px;height:22px;position:absolute;top:50%;left:10px;margin-top:-11px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=11.$=10.$icon"></div></div><div style="position:absolute;left:40px;right:40px;top:0px;bottom:0px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=11.$/=12"><input type="text" style="width:100%;height:100%;font-size:14px;font-family:mainnext_regular;background-color:transparent;color:#ffffff;" placeholder="SEARCH ARTISTS, TRACKS, ALBUMS" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=11.$/=12.0"/></div></div><div style="background-color:rgba(27,27,27,0.92);height:10px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=11.$/=12"></div></div><div style="position:absolute;top:0px;left:620px;right:0px;background-color:rgba(27,27,27,0.92);height:60px;line-height:60px;overflow:hidden;min-width:500px;padding-left:10px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12"><div style="position:absolute;top:0px;bottom:0px;right:0px;width:357px;padding-left:141px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0"><a class="" href="/import" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$/=10"><div style="padding-left:40px;position:absolute;left:0px;top:10px;bottom:10px;cursor:pointer;line-height:40px;color:rgb(255,255,255);" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$/=10.$import"><div style="position:absolute;top:0;bottom:0;left:0;right:0;padding:0;margin:0;display:flex;align-items:center;justify-content:center;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$/=10.$import.$=10"><div style="background-image:url(&#x27;/assets/3bec3e57cb5ee05658440d21984fb7b7.png&#x27;);background-repeat:no-repeat;background-position:0px -194px;width:28px;height:28px;position:absolute;top:50%;left:0px;margin-top:-14px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$/=10.$import.$=10.$icon"></div></div><span data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$/=10.$import.1">Import Playlists</span></div></a><div style="margin-left:10px;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$admin/=1$admin"><div style="cursor:pointer;float:left;" data-reactid=".1hkqsbm9n9c.0.$/=11.$/=12.0.$admin/=1$admin.$login">Login</div></div></div></div></div></div><noscript data-reactid=".1hkqsbm9n9c.1"></noscript></div></div> <div class="___iso-state___" data-key="_0" data-meta="{}" data-state="&quot;{/&quot;UserStore/&quot;:{/&quot;user/&quot;:{/&quot;authenticated/&quot;:false,/&quot;isWaiting/&quot;:false}},/&quot;SearchStore/&quot;:{/&quot;focused/&quot;:false,/&quot;input/&quot;:/&quot;/&quot;,/&quot;timeout/&quot;:null,/&quot;searchRequests/&quot;:[],/&quot;artists/&quot;:null,/&quot;artistsFailed/&quot;:false,/&quot;artistsLoading/&quot;:false,/&quot;tracks/&quot;:null,/&quot;tracksFailed/&quot;:false,/&quot;tracksLoading/&quot;:false,/&quot;albums/&quot;:null,/&quot;albumsFailed/&quot;:false,/&quot;albumsLoading/&quot;:false,/&quot;playlists/&quot;:null,/&quot;playlistsFailed/&quot;:false,/&quot;playlistsLoading/&quot;:false,/&quot;youtubes/&quot;:null,/&quot;youtubesFailed/&quot;:false,/&quot;youtubesLoading/&quot;:false,/&quot;soundclouds/&quot;:null,/&quot;soundcloudsFailed/&quot;:false,/&quot;soundcloudsLoading/&quot;:false},/&quot;PlayerStore/&quot;:{/&quot;player/&quot;:null,/&quot;playerSecond/&quot;:null,/&quot;playingTrack/&quot;:null,/&quot;playingTrackSecond/&quot;:null,/&quot;videoId/&quot;:null,/&quot;videoIdSecond/&quot;:null,/&quot;makingPlayingTrackPlayable/&quot;:false,/&quot;radio/&quot;:false,/&quot;startSeconds/&quot;:0,/&quot;current/&quot;:0,/&quot;total/&quot;:0,/&quot;perc/&quot;:0,/&quot;currentSecond/&quot;:0,/&quot;totalSecond/&quot;:0,/&quot;percSecond/&quot;:0,/&quot;playing/&quot;:false,/&quot;playingSecond/&quot;:false,/&quot;secondsListened/&quot;:0,/&quot;secondsListenedSecond/&quot;:0,/&quot;expand/&quot;:false,/&quot;source/&quot;:null,/&quot;tracksQueue/&quot;:[],/&quot;tracksPrevQueue/&quot;:[],/&quot;favorite/&quot;:false,/&quot;random/&quot;:false,/&quot;repeat/&quot;:false,/&quot;mute/&quot;:false,/&quot;volume/&quot;:100,/&quot;mode/&quot;:/&quot;standard/&quot;},/&quot;ImportStore/&quot;:{/&quot;url/&quot;:/&quot;/&quot;,/&quot;error/&quot;:false,/&quot;focused/&quot;:false,/&quot;loading/&quot;:false,/&quot;loaded/&quot;:false,/&quot;playlist/&quot;:null}}&quot;"></div> </div> <!-- Google Analytics: change UA-XXXXX-X to be your site''s ID --> <!-- <script> (function(i,s,o,g,r,a,m){i[''GoogleAnalyticsObject'']=r;i[r]=i[r]||function(){ (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o), m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m) })(window,document,''script'',''//www.google-analytics.com/analytics.js'',''ga''); ga(''create'', ''UA-XXXXX-X'', ''auto''); ga(''send'', ''pageview''); </script> --> <script src="https://cdnjs.cloudflare.com/ajax/libs/fastclick/1.0.3/fastclick.min.js"></script> <script type="text/javascript"> if (''addEventListener'' in document) { document.addEventListener(''DOMContentLoaded'', function() { FastClick.attach(document.body); }, false); } </script> <script type="text/javascript" charset="utf-8" src="/assets/app.js"></script> </body> </html>

Este es mi server.jsx:

import Iso from ''iso''; import React from ''react''; import ReactDomServer from ''react-dom/server''; import { RoutingContext, match } from ''react-router'' import createLocation from ''history/lib/createLocation''; import alt from ''altInstance''; import routes from ''routes.jsx''; import html from ''base.html''; /* * @param {AltObject} an instance of the Alt object * @param {ReactObject} routes specified in react-router * @param {Object} Data to bootstrap our altStores with * @param {Object} req passed from Express/Koa server */ const renderToMarkup = (alt, state, req, res) => { let markup, content; let location = new createLocation(req.url); alt.bootstrap(state); match({ routes, location }, (error, redirectLocation, renderProps) => { if (redirectLocation) res.redirect(301, redirectLocation.pathname + redirectLocation.search) else if (error) res.status(500).send(error.message) else if (renderProps == null) res.status(404).send(''Not found'') else content = ReactDomServer.renderToString(<RoutingContext {...renderProps} />); markup = Iso.render(content, alt.flush()); }); return markup; }; /* * Export render function to be used in server/config/routes.js * We grab the state passed in from the server and the req object from Express/Koa * and pass it into the Router.run function. */ export default function render(state, req, res) { const markup = renderToMarkup(alt, state, req, res); return html.replace(''CONTENT'', markup); };

Y este es mi cliente.jsx:

import React from ''react''; import ReactDOM from ''react-dom''; import Iso from ''iso''; import createBrowserHistory from ''history/lib/createBrowserHistory''; import { Router } from ''react-router''; import alt from ''altInstance''; import routes from ''routes.jsx''; /* * Client side bootstrap with iso and alt */ Iso.bootstrap((state, _, container) => { alt.bootstrap(state); ReactDOM.render(<Router history={createBrowserHistory()} children={routes} />, container); });

Y mi routes.jsx:

import React from ''react''; import Route from ''react-router''; import App from ''components/App''; import ImportPlaylist from ''components/ImportPlaylist''; import Login from ''components/Login''; import Logout from ''components/Logout''; import Player from ''components/Player/Player''; import Test from ''components/Test''; export default ( <Route path="/" component={App}> <Route path="login" component={Login} /> <Route path="logout" component={Logout} /> <Route name="test" path="test" component={Test} /> <Route name="import" path="import" component={ImportPlaylist} /> <Route name="player" path="/:playlist" component={Player} /> </Route> );


Me encontré con este problema en una aplicación isomórfica en la que estaba trabajando. Lo que funcionó para mí es, lo creas o no, vaciar el caché y volver a cargar la aplicación en Chrome. Parecía que el viejo DOM estaba almacenado en caché de algún modo en el navegador :)


Para aquellos que buscan en Google y vienen aquí, una forma extraña de terminar con este problema es cuando ni siquiera está utilizando la representación isomórfica (es decir, no representa nada en el lado del servidor). Esto me sucedió cuando HtmlWebpackPlugin una plantilla con HtmlWebpackPlugin para procesar un archivo index.html .

En mi archivo index.html ya bundle.js archivo bundle.js , y el complemento anterior también incluye otro bundle.js través de un script-src. Asegúrate de configurar inject: false en tu constructor HtmlWebpackPlugin .


Para mí matar nodejs por completo y reiniciar el trabajo


Si está procesando su contenido principal dentro de un componente de diseño, deberá renderizar el diseño como marcado estático (sin atributos de reacción) para que la suma de verificación del contenido coincida entre el cliente y el servidor.

Servidor:

app.get(''/'', (req, res) => { // render the content to a string so it has the same checksum as on the client // render the layout to static markup so that it does affect the checksum // this ensures that rendering is isomorphic and client doesn''t override server markup const content = reactDomServer.renderToString(<MyContent />) const html = ''<!DOCTYPE html>'' + reactDomServer.renderToStaticMarkup(<HtmlLayout content={content} />) res.send(html) })

HtmlLayout:

export default class HtmlLayout extends React.Component<any, any> { public render () { return ( <html lang=''en''> <head> <meta charSet=''utf-8'' /> <meta name=''viewport'' content=''width=device-width, initial-scale=1'' /> <title>Untitled</title> <link rel=''stylesheet'' href=''/style/bundle.css'' /> </head> <body> { /* insert the content as a string so that it can be rendered separate with its own checksum for proper server-side rendering */ } <div id=''content'' dangerouslySetInnerHTML={ {__html: this.props.content} } /> <script src=''scripts/bundle.js''></script> </body> </html> ) } }

Cliente:

const root = document.getElementById(''content'') DOM.render(<MyContent />, root)

Referencia: http://jeffhandley.github.io/QuickReactions/20-final-cleanup


ADVERTENCIA La respuesta popular aquí no es correcta. Lo que está haciendo es eliminar completamente el DOM existente y reemplazarlo con un renderizado nuevo en el cliente. Significa que pierdes el rápido renderizado superficial de React y estás perdiendo perf, y como consecuencia también se traga el error de OP, y cualquier otro error que puedas tener.

El problema parece ser con CSS, si está utilizando autoprefixer y estilos en línea que expliquen su diferencia aquí.

El lado del servidor ha renderizado align-items: center y el cliente se ha dado cuenta de que está en un navegador webkit y lo ha prefijado automáticamente para que -webkit-align-items.

Por favor publique más información sobre la configuración de CSS, y cualquier componente que use estilos en línea o similar.


Nota: Esto se aplica a las versiones anteriores de React. Si usa React 16, debe usar ReactDOM.hydrate()

Además , la sugerencia a continuación dará como resultado una repetición del lado del cliente, como lo sugiere una de las siguientes respuestas.

Esto puede sonar increíblemente simple, pero en su plantilla del lado del servidor, ajuste su marcado Reaccionar en un <div> adicional:

<!-- hypothetical handlebars template --> <section role="main" class="react-container"><div>{{{reactMarkup}}}</div></section>

¿Por qué funciona esto? En el cliente, React tiene una tendencia a ajustar su representación de su componente raíz con un div superfluo. ReactDOMServer.render no parece comportarse de esta manera, por lo tanto, cuando se procesa isomórficamente en el mismo contenedor, la suma de verificación Adler-32 de su DOM difiere.