Estoy tratando de conseguir un isomorfo Node.js Expresa, Webpack, Reaccionan de la aplicación de trabajo. Me da 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 está siendo enviado por el servidor al navegador inicialmente:

<!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 servidor.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 mis rutas.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>
);

7 Comentarios

  1. 109

    Nota: Esto se aplica a las versiones anteriores de Reaccionar. Si usted está usando Reaccionar 16, usted debe utilizar ReactDOM.hidratar()

    También, la siguiente sugerencia resultará en un lado del cliente vuelva a representar, según lo sugerido por una de las siguientes respuestas.


    Esto puede sonar loco simple, pero en su plantilla de servidor, envuelva el Reaccionar de marcado en un extra de <div>:

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

    ¿Por qué funciona esto? En el cliente, Reaccionan tiene una tendencia a envolver su representación de la componente de la raíz con una superfluo div. ReactDOMServer.render no parece que se comportan de esta manera, por lo tanto cuando uno hace en el mismo contenedor isomorphically, el Adler-32 de la suma de comprobación de su DOM diferencia.

    • No me queda claro en qué lado del servidor de la plantilla es en esta instancia. He incluido mi lado cliente y servidor de renderizado de código de arriba en la pregunta, así como mi archivo de rutas.
    • Aah, estás usando el Iso módulo y que está regresando a HTML estático. Lo siento, pero no voy a ser capaz de ayudar a usted específicamente como tengo poca experiencia con este tipo de configuración, pero mi respuesta, me ha funcionado a través de numerosos proyectos cuando se utiliza un Expreso motor de plantilla y de paso a mi en el lado del servidor Reaccionar de representación como una vista de la propiedad.
    • A mí me funciona! Gracias @JamesWright
    • Funcionó a la perfección gracias @JamesWright!
    • Wow esta respuesta tuvo manera de encontrar! Esto funciona para mí también!
    • Después de una suerte de google de golpe, esta probablemente me salvó horas de cachondeo. Gracias +1
    • Gracias @JamesWright, esta solución ayuda.
    • Increíble… Gracias @JamesWright
    • Gracias @JamesWright. Estoy representación de mi componente de la raíz usando react-dom/renderToString y de insertarlo en el documento utilizando dangerouslySetInnerHTML={{ __html: content }}, pero esta solución todavía funcionaba cuando yo envuelto content en un elemento div vacío.
    • Gracias me estaba golpeando mi cabeza porque yo era la representación dentro de un div que contenía en sí el contenido antes de
    • Muchas gracias @JamesWright! Facebook del equipo – hay un tema abierto para esto en github?
    • Precioso, que lo hizo.
    • Que funcionó sorprendentemente para mí también. Convincible explicación.
    • Esto funciona, pero al mismo tiempo parece un poco loco. ¿No debería esto ser documentados/fija en algún lugar?
    • Esta no es la solución. Lo que esto hace es quitar el lado del cliente superficial de procesamiento, y en lugar de realizar un mucho más lento de procesamiento, la eliminación de los existentes DOM completamente, y reemplazarlo con el nuevo resultado. Como consecuencia de ello también se traga este error, y cualquier otro de los errores que pueda tener.
    • Se la especificación de un elemento primario de todo el lado del servidor Reaccionar de representación causar eso? Sin duda, este enfoque de evitar una re-render como ahora hay paridad entre el cliente y el servidor árbol DOM. Si este no es el caso, sin embargo, podría enlazar a una fuente para confirmar esto? Todavía no estoy capaz de encontrar otras soluciones a este problema un año después de que envió esta respuesta. Gracias!
    • jaja esta extraña forma en que funciona bien 😀

  2. 17

    Para aquellos googlear y venir aquí, de una extraña manera de acabar con este problema es cuando ni siquiera estás usando isomorfo de representación (es decir, no rendir nada en el lado del servidor). Esto sucedió a mí cuando se utiliza una plantilla con el HtmlWebpackPlugin a un proceso de index.html archivo.

    En mi index.html archivo ya he incluido el bundle.js archivo de mí mismo, y el anterior plugin también incluye otro bundle.js a través de un script src. Asegúrese de que usted está estableciendo inject: false a su HtmlWebpackPlugin constructor.

    • En el clavo! Estoy usando Rails, y mi liado archivo js estaba siendo recogido dos veces. Una vez cuando yo la incluí en mi página de la plantilla, y también con require_tree . en application.js. He quitado el require_tree . en línea application.js, pero que puede no ser la mejor solución para todos.
  3. 13

    Para mí matar a nodejs completamente y reiniciar trabajado

    • trabajó para mí también, pero todavía me pregunto ¿cuál es la causa raíz y cómo prevenir este problema
    • a mí también, esto suena raro pero FUNCIONA
    • El problema puede ser que usted no tiene bañera de recarga en el servidor, pero no en el cliente. Así, cuando se hacen cambios, el código de cliente es hasta la fecha, pero el código del servidor no es, y que producen resultados diferentes. El reinicio del equipo asegurarse de que el cliente y el servidor están utilizando el mismo código (hasta que cambie de nuevo)
  4. 8

    ADVERTENCIA La respuesta popular aquí no es la correcta. Lo que hace es quitar el DOM existente en su totalidad, y reemplazarlo con un nuevo procesamiento en el cliente. Esto significa que usted pierde el rápido superficial de procesamiento de Reaccionar y de desperdicio de perf, y como consecuencia también se traga el OP de error, y cualquier otro de los errores que pueda tener.

    El problema parece ser con CSS – si usted está usando autoprefixer y en línea-los estilos que explicaría su diferencia aquí.

    El lado del servidor se ha dictado alinear elementos:el centro y el cliente se dio cuenta de que en un navegador webkit y automáticamente el prefijo para usted -webkit-align-elementos.

    Por favor enviar más información acerca de la CSS configurar, y cualquier componente de uso de línea de estilos o similar.

    • ¿Tiene una fuente con la que se puede respaldar por ejemplo, un archivo en Reaccionar el código fuente? No puedo encontrar nada para sugerir esto, además de que espero que el marcado de la reutilización de advertencia se emite incluso cuando la inicial de render se pierde. No estoy actualmente convencido de que es tragar el error.
    • Lo siento, no he iniciado sesión en un largo tiempo. imagen: imgur.com/a/bbOFr código: github.com/facebook/react/blob/… Si usted ajusta el marcado en un div que usted encontrará que ‘shouldReuseMarkup» es falso.
    • Wow, así que sin duda se adentró profundamente! Gracias por señalarlo. No voy a quitar mi respuesta para la posteridad del bien, pero yo sin duda especificar esta información como una advertencia.
  5. 6

    Si usted presta su principal contenido dentro de un esquema de componente, deberá representar el diseño estático de marcado (no reaccionar atributos), de modo que la suma de los contenidos de los encuentros 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

  6. 2

    En mi caso el problema fue causado por el hecho de que yo estaba usando MediaQuery componente de «reaccionar sensibles’ sin pasar por el «valor» de la propiedad que es utilizada por el componente cuando no es capaz de acceder a la anchura de la pantalla (por ejemplo, en el Servidor).

  7. 1

    Me encontré con este problema en un Isomorfo aplicación en el que estaba trabajando. Lo que funcionó para mí es, lo creas o no, el Vaciado de la caché y duro a cargar la aplicación en Chrome. Parecía que el viejo DOM fue de alguna manera en caché en el Navegador 🙂

    • No funciona para mí. ¿Alguien realmente entiende este error?

Dejar respuesta

Please enter your comment!
Please enter your name here