Skip to content

10. Exemplo [nuxt-07]: os contextos do cliente e do servidor

10.1. Présentation

O exemplo [nuxt-07] tem como objetivo explorar o objeto [context] tanto do lado do servidor como do lado do cliente. Não se deve esquecer que estes dois intervenientes nas aplicações [nuxt] estão separados: não partilham nada, exceto:

  • o que o servidor decide enviar ao cliente (na resposta HTTP e na página enviada);
  • o que o cliente decide enviar ao servidor (na sua solicitação HTTP);

Portanto, se, como veremos, os objetos manipulados pelo servidor e os manipulados pelo cliente tiverem o mesmo nome, não são idênticos: podem, por vezes, ser cópias um do outro, mas nunca têm a mesma referência. Alterar um objeto do cliente não tem qualquer efeito sobre o objeto com o mesmo nome, do lado do servidor, e vice-versa.

O exemplo [nuxt-07] é obtido inicialmente por cópia do exemplo [nuxt-01]:

Image

  • em [2], vamos adicionar um plugin comum ao cliente e ao servidor;
  • em [3], iremos alterar ligeiramente a página [index];

10.2. O plugin [common / main.js]

O plugin [common / main.js] é executado tanto pelo cliente como pelo servidor: isto deve-se à seguinte configuração [nuxt.config.js]:


/*
   ** Plugins to load before mounting the App
   */
plugins: [{ src: '~/plugins/common/main.js' }],
  • linha 4: a ausência da propriedade [mode] faz com que o plugin [~/plugins/common/main.js] seja executado tanto pelo cliente como pelo servidor: primeiro pelo servidor, depois pelo cliente;

Este plugin será o seguinte:


/* eslint-disable no-undef */
/* eslint-disable no-console */
export default function(...args) {
  // quem executa este código?
  console.log('[main server], process.server=', process.server, 'process.client=', process.client)
  const who = process.server ? 'server' : 'client'
  const main = '[main ' + who + ']'
  // número de argumentos
  console.log(main + ', il y a', args.length, 'arguments')

  // primeiro argumento
  const context = args[0]
  // chaves do contexto
  dumpkeys(main + ', context', context)
  // a aplicação
  dumpkeys(main + ', context.app', context.app)
  // a rota
  dumpkeys(main + ', context.route', context.route)
  console.log(main + ', context.route=', context.route)
  // o router
  dumpkeys(main + ', context.app.router', context.app.router)
  // o router.options.routes
  dumpkeys(main + ', context.app.router.options.routes', context.app.router.options.routes)
  console.log(main + ', context.app.router.options.routes=', context.app.router.options.routes)

  // 2.º argumento
  const inject = args[1]
  console.log('inject=', typeof inject)
}

function dumpkeys(message, object) {
  // lista de chaves de [object]
  const ligne = 'Liste des clés [' + message + ']'
  console.log(ligne)
  // lista de chaves
  if (object) {
    console.log(Object.keys(object))
  }
}
  • linhas 31-39: a função [dumpkeys] lista as propriedades do objeto passado como segundo parâmetro. Esta lista é precedida pela mensagem passada como primeiro parâmetro;
  • linha 3: queremos saber quantos argumentos a função recebe. Para isso, utilizamos a notação [...args], que terá como efeito colocar os parâmetros efetivos da função na tabela [args]. Vamos descobrir que existem dois argumentos;
  • linha 5: exibimos quem está a executar o código, se é o servidor ou o cliente;
  • linha 6: quem executa o código, o cliente ou o servidor;
  • linha 7: uma constante de cadeia de caracteres utilizada nos registos;
  • linha 12: vamos descobrir que o primeiro argumento recebido pelo plugin é o contexto do executor;
  • linha 14: lista das chaves do objeto [context];
  • linha 16: vamos descobrir que o objeto [context] tem uma propriedade [app] que representa a aplicação [nuxt];
  • linha 18: vamos descobrir que o objeto [context] tem uma propriedade [route] que representa a rota atual do router;
  • linha 21: vamos descobrir que o objeto [app] tem uma propriedade [router] que representa o router;
  • linha 23: o objeto [router.options.routes] representa as diferentes rotas da aplicação;
  • linhas 27-28: o segundo argumento do plugin é a função [inject] que utilizámos no exemplo [nuxt-06];

10.3. O plugin executado pelo servidor

Ao ser executado, o servidor apresenta o seguinte:

[main server], process.server= true process.client= false
[main server], il y a 2 arguments
Liste des clés [[main server], context]
[
  'isStatic',
  'isDev',
  'isHMR',
  'app',
  'payload',
  'error',
  'base',
  'env',
  'req',
  'res',
  'ssrContext',
  'redirect',
  'beforeNuxtRender',
  'route',
  'next',
  '_redirected',
  '_errored',
  'params',
  'query',
   '$axios'
]
  • linhas 11-12: já tivemos oportunidade de utilizar as propriedades [base] e [env], cujos valores provêm do ficheiro [nuxt.config.js];
  • linha 8: a propriedade [app] designa a aplicação [nuxt];
  • linha 18: a propriedade [route] indica a rota atual do router, ou seja, a página que o servidor irá enviar;
  • linha 13: o pedido HTTP do navegador do cliente;
  • linha 14: a resposta HTTP do servidor;

A lista de propriedades de [context.app] é a seguinte:

Liste des clés [[main server], context.app]
[
  'router',
  'nuxt',
  'head',
  'render',
  'data',
  'beforeCreate',
  'created',
  'mounted',
  'watch',
  'computed',
  'methods',
  'components',
  'context',
   '$axios'
]
  • linha 3: a propriedade [router] dá-nos acesso ao router da aplicação. Isto é importante no [nuxt], uma vez que o router é definido pelo próprio [nuxt] e não pelo programador. Esta propriedade constitui um acesso concedido ao programador para alterar o router;

A lista de propriedades de [context.route] é a seguinte:

Liste des clés [[main server], context.route]
[
  'name',
  'meta',
  'path',
  'hash',
  'query',
  'params',
  'fullPath',
   'correspondência'
]

A rota [context.route] no arranque do servidor é a seguinte:

[main server], context.route= {
  name: 'index',
  meta: [
    {}
  ],
  path: '/',
  hash: '',
  query: {},
  params: {},
  fullPath: '/',
  matched: [
    {
      path: '',
      regex: /^(?:\/(?=$))?$/i,
      components: [Object],
      instances: {},
      name: 'index',
      parent: undefined,
      matchAs: undefined,
      redirect: undefined,
      beforeEnter: undefined,
      meta: {},
      props: {}
    }
  ]
}
  • linha 2: verifica-se que a página seguinte do servidor é [index] e que o seu caminho é [/] (linha 10);
  • linha 22: a propriedade [meta] permite adicionar propriedades às rotas;

As propriedades do router [context.app.router] do servidor são as seguintes:

Liste des clés [[main server], context.app.router]
[
  'app',
  'apps',
  'options',
  'beforeHooks',
  'resolveHooks',
  'afterHooks',
  'matcher',
  'fallback',
  'mode',
   'histórico'
]

É na propriedade [context.app.router.options.routes] que se encontram as diferentes rotas da aplicação:

Liste des clés [[main server], context.app.router.options.routes]
[
  '0',
  '1',
  '2'
]
[main server], context.app.router.options.routes= [
  {
    path: '/page1',
    component: [Function: _d7b6c762],
    name: 'page1'
  },
  {
    path: '/page2',
    component: [Function: _d79a9860],
    name: 'page2'
  },
  {
    path: '/',
    component: [Function: _31eaad9f],
    name: 'index'
  }
]

Por fim, o segundo argumento:

inject= function

10.4. A página [index] do servidor

A página [index] é a seguinte:


<!-- página principal -->
<template>
  <Layout :left="true" :right="true">
    <!-- navegação -->
    <Navigation slot="left" />
    <!-- mensagem-->
    <b-alert slot="right" show variant="warning">
      Home
    </b-alert>
  </Layout>
</template>

<script>
/* eslint-disable no-undef */
/* eslint-disable no-console */
/* eslint-disable nuxt/no-env-in-hooks */

import Navigation from '@/components/navigation'
import Layout from '@/components/layout'

export default {
  name: 'Home',
  // componentes utilizados
  components: {
    Layout,
    Navigation
  },
  data() {
    return {
      who: process.server ? 'server' : 'client'
    }
  },

  // ciclo de vida
  beforeCreate() {
    // cliente e servidor
    console.log('[home beforeCreate]')
  },
  created() {
    // cliente e servidor
    console.log('[home ' + this.who + ' created]')
    this.dumpkeys('[home ' + this.who + ' created], this.$nuxt', this.$nuxt)
    this.dumpkeys('[home ' + this.who + ' created], this.$nuxt.context', this.$nuxt.context)
  },
  beforeMount() {
    // apenas cliente
    console.log('[home ' + this.who + ' beforeMount]')
  },
  mounted() {
    // apenas cliente
    console.log('[home ' + this.who + ' mounted]')
  },
  methods: {
    dumpkeys(message, object) {
      // lista de chaves de [object]
      const ligne = 'Liste des clés [' + message + ']'
      console.log(ligne)
      if (object) {
        console.log(Object.keys(object))
      }
    }
  }
}
</script>
  • linha 43: já vimos que o contexto de uma página pode ser encontrado em [this.$nuxt.context];
  • linha 42: também são apresentadas as propriedades do objeto [this.$nuxt];

Executada pelo servidor, esta página gera os seguintes registos:

[home beforeCreate]
[home server created]
Liste des clés [[home server created], this.$nuxt]
[
  '_uid',
  '_isVue',
   '$options',
  '_renderProxy',
  '_self',
   '$parent',
   '$root',
   '$children',
   '$refs',
  '_watcher',
  '_inactive',
  '_directInactive',
  '_isMounted',
  '_isDestroyed',
  '_isBeingDestroyed',
  '_events',
  '_hasHookEvent',
  '_vnode',
  '_staticTrees',
   '$vnode',
   '$slots',
   '$scopedSlots',
  '_c',
   '$createElement',
   '$attrs',
   '$listeners',
  '_routerRoot',
  '_router',
  '_route',
  '_bv__modal',
  '_bv__toast',
  '_vueMeta',
  'nuxt',
  '_watchers',
  'refreshOnlineStatus',
  'refresh',
  'errorChanged',
  'setLayout',
  'loadLayout',
  '_data',
  'layoutName',
  'layout',
  'isOnline',
  '_computedWatchers',
  'isOffline',
  'error',
   'contexto'
]

As propriedades do contexto do servidor na página [index] são as seguintes:

Liste des clés [[home server created], this.$nuxt.context]
[
  'isStatic',
  'isDev',
  'isHMR',
  'app',
  'payload',
  'error',
  'base',
  'env',
  'req',
  'res',
  'ssrContext',
  'redirect',
  'beforeNuxtRender',
  'route',
  'next',
  '_redirected',
  '_errored',
  'params',
  'query',
   '$axios'
]

São as mesmas propriedades que no objeto [context] do plugin.

10.5. O plugin executado pelo cliente

Assim que o servidor enviar a página [index] para o navegador do cliente, os scripts do cliente assumem o controlo. O plugin [main.js] será então executado. Estes são os registos:

[main server], process.server= false process.client= true
[main client], il y a 2 arguments
Liste des clés [[main client], context]
Array(17)0: "isStatic"1: "isDev"2: "isHMR"3: "app"4: "payload"5: "error"6: "base"7: "env"8: "redirect"9: "nuxtState"10: "route"11: "next"12: "_redirected"13: "_errored"14: "params"15: "query"16: "$axios"length: 17__proto__: Array(0)
Liste des clés [[main client], context.app]
Array(14)0: "router"1: "nuxt"2: "head"3: "render"4: "data"5: "beforeCreate"6: "created"7: "mounted"8: "watch"9: "computed"10: "methods"11: "components"12: "context"13: "$axios"length: 14__proto__: Array(0)
Liste des clés [[main client], context.route]
Array(8)0: "name"1: "meta"2: "path"3: "hash"4: "query"5: "params"6: "fullPath"7: "matched"length: 8__proto__: Array(0)
[main client], context.route= ObjectfullPath: "/"hash: ""matched: [{…}]meta: [{…}]name: "index"params: {}path: "/"query: {}__proto__: Object
Liste des clés [[main client], context.app.router]
Array(10)0: "app"1: "apps"2: "options"3: "beforeHooks"4: "resolveHooks"5: "afterHooks"6: "matcher"7: "fallback"8: "mode"9: "history"length: 10__proto__: Array(0)
Liste des clés [[main client], context.app.router.options.routes]
Array(3)0: "0"1: "1"2: "2"length: 3__proto__: Array(0)
[main client], context.app.router.options.routes= Array(3)0: {path: "/page1", name: "page1", component: ƒ}1: {path: "/page2", name: "page2", component: ƒ}2: {path: "/", name: "index", component: ƒ}length: 3__proto__: Array(0)
inject= function

Encontramos propriedades semelhantes às encontradas no lado do servidor, com algumas propriedades que desapareceram e outras que surgiram. Assim, na linha 4, não se encontram as propriedades [req, res], que correspondiam às solicitações HTTP do navegador do cliente e à resposta HTTP do servidor.

10.6. A página [index] do cliente

A página [index] do cliente gera os seguintes registos:

1
2
3
4
5
6
7
8
[home beforeCreate]
[home client created]
Liste des clés [[home client created], this.$nuxt]
(51) ["_uid", "_isVue", "$options", "_renderProxy", "_self", "$parent", "$root", "$children", "$refs", "_watcher", "_inactive", "_directInactive", "_isMounted", "_isDestroyed", "_isBeingDestroyed", "_events", "_hasHookEvent", "_vnode", "_staticTrees", "$vnode", "$slots", "$scopedSlots", "_c", "$createElement", "$attrs", "$listeners", "_routerRoot", "_router", "_route", "_bv__modal", "_bv__toast", "_vueMeta", "nuxt", "_watchers", "refreshOnlineStatus", "refresh", "errorChanged", "setLayout", "loadLayout", "_data", "layoutName", "layout", "isOnline", "_computedWatchers", "isOffline", "error", "context", "_name", "setTransitions", "$loading", "$el"]
Liste des clés [[home client created], this.$nuxt.context]
(17) ["isStatic", "isDev", "isHMR", "app", "payload", "error", "base", "env", "redirect", "nuxtState", "route", "next", "_redirected", "_errored", "params", "query", "$axios"]
[home client beforeMount]
[home client mounted]
  • linha 4: as propriedades do objeto [this.$nuxt]. Trata-se de um objeto complexo com 51 propriedades;
  • linha 6: as propriedades do objeto [this.$nuxt.context]. Encontram-se as mesmas propriedades que no objeto [context] do plugin do cliente;