nuxt 在本地为下一个 JS 身份验证重置密码生成工作正常,但在部署到 plesk 服务器时却不行

nuxt generate working ok locally for next JS auth reset-password but not when deployed to plesk server

我有一个 Nuxt JS 应用设置要使用 Nuxt Auth。 这在本地工作正常。

具体来说,我正在生成一封发送给用户的电子邮件,其中包含 link 以重置其表单密码

http://localhost:3000/reset-password/ca62c3554c8058c9ddf11b709fc451405ffa99f4b22a88d84e087f5b40fb6d1f

当他们点击它时 - 它被解析 JWT 的 nuxt 路由拾取。 我在本地使用 nuxt start 服务它——我相信它从 dist 目录服务,所以应该是静态服务的一个很好的测试

当我将它部署到远程 lightsail 服务器 运行ning Ubuntu 和 Plesk、Nginx 和 Apache 时,我使用 nuxt generate 部署它,并将生成的 dist 目录的内容复制到 httpdocs 目录. 当遵循相同的工作流程并且用户单击 link 时,它不会被 nuxt 生成的静态 html 文件之一捕获,我得到 404。 所有其他 nuxt 路由都正在生成到文件中。 我错过了什么?

nuxt.config.js

export default {
  target: 'static',
  loading: {
    color: '#3700b3',
    height: '5px',
  },
  env: {
    apiUrl: process.env.NODE_ENV === 'production' ? process.env.PLATFORM_API_URL : 'http://localhost:8000',
    mainUrl: process.env.NODE_ENV === 'production' ? process.env.PLATFORM_URL : 'http://localhost:3000',
    googleSiteKey: process.env.RECAPTCHA_SITE_KEY || '',
  },
  ssr: false,
  head: {
    titleTemplate: `%s - ${process.env.PLATFORM_NAME || 'Some platform name'}`,
    title: process.env.PLATFORM_NAME || 'Some platform name',
    meta: [
      { charset: 'utf-8' },
      { name: 'viewport', content: 'width=device-width, initial-scale=1.0' },
      { hid: 'description', name: 'description', content: 'Virtua Centre' },
    ],
    link: [{ rel: 'icon', type: 'image/x-icon', href: '/favicon.ico' }],
    script: [
      {
        src: 'https://platform.twitter.com/widgets.js',
      },
      {
        src: 'https://js.stripe.com/v3',
      },
    ],
  },
  plugins: [
    {
      src: '@/plugins/vue-page-transition.js',
    },
    {
      src: '@/plugins/plateform-detector.js',
    },
    { src: '~/plugins/TiptapVuetify', mode: 'client' },
    { src: '@/plugins/filters.js' },
    { src: '~/plugins/i18n.js' },
    { src: '~/plugins/locales.js' },
  ],
  components: true,
  buildModules: [
    '@nuxtjs/eslint-module',
    '@nuxtjs/stylelint-module',
    ['@nuxtjs/vuetify'],
    '@nuxtjs/date-fns',
  ],
  modules: [
    'nuxt-i18n',
    '@nuxtjs/axios',
    '@nuxtjs/auth-next',
    ['v-currency-field/nuxt-treeshaking'],
    'vue-currency-filter/nuxt',
    'vuetify-dialog/nuxt',
  ],
  i18n: {
    strategy: 'no_prefix',
    locales: [
      {
        code: 'en',
        name: 'English',
        file: 'en-US.js',
        flag: '/flag-icon/flags/1x1/us.svg',
      },
      {
        code: 'kk',
        name: 'Kazakh',
        file: 'en-KK.js',
        flag: '/flag-icon/flags/1x1/kz.svg',
      },
      {
        code: 'ru',
        name: 'Russian',
        file: 'en-RU.js',
        flag: '/flag-icon/flags/1x1/ru.svg',
      },
    ],
    lazy: true,
    langDir: 'lang',
    defaultLocale: 'en',
    vueI18n: {
      fallbackLocale: 'en',
    },
  },
  axios: {
    credentials: true,
    baseURL: process.env.NODE_ENV === 'production' ? process.env.PLATFORM_API_URL : 'http://localhost:8000',
  },
  auth: {
    redirect: {
      login: '/login',
      logout: false,
      callback: '/',
      home: false,
    },
    strategies: {
      local: {
        token: {
          property: 'data.access_token',
          maxAge: 36000,
        },
        user: {
          property: 'data',
        },
        endpoints: {
          login: { url: '/auth/login', method: 'post' },
          logout: { url: '/logout', method: 'post' },
          user: { url: '/me', method: 'get' },
        },
      },
    },
  },
  vue: {
    config: {
      productionTip: false,
      devtools: true,
    },
  },
  vuetify: {
    theme: {
      themes: {
        light: {
          primary: '#4F91FF',
          secondary: '#00109c',
          success: '#00B485',
          lsmbutton: '#FFBF42',
          error: '#F85032',
        },
      },
    },
  },
  build: {
    extractCSS: true,
    transpile: ['vuetify/lib', 'tiptap-vuetify', 'vee-validate/dist/rules'],
    babel: {
      plugins: [['@babel/plugin-proposal-private-methods', { loose: true }]],
    },
    extend(config, ctx) {
      config.module.rules.push({
        test: /\.(ogg|mp3|wav|mpe?g)$/i,
        loader: 'file-loader',
        options: {
          name: '[path][name].[ext]',
        },
      })
    },
    splitChunks: {
      layouts: true,
    },
  },
}

package.json 的脚本部分

"scripts": {
  "dev": "nuxt --hostname 127.0.0.1 --port 3000",
  "build": "nuxt build",
  "start": "nuxt start",
  "generate": "nuxt generate",
  "lint:js": "eslint --ext .js,.vue --ignore-path .gitignore .",
  "lintfix": "eslint --fix --ext .vue --ignore-path .gitignore .",
  "lint:style": "stylelint **/*.{vue,css} --ignore-path .gitignore",
  "lint": "npm run lint:js && npm run lint:style",
  "test": "jest"
},

我正在使用 npm

阅读周围我可以看到处理动态路由的标准解决方案是更新 nuxt.config.js generate.routes 部分中的配置。如 this medium article

中详述

这似乎可以通过在生成时从服务器获取所有值来实现。我认为这不适用于身份验证令牌,因为用户可以随时注册 - 特别是在 nuxt 生成 运行.

之后

重置密码功能

index.vue

    <template>
      <div v-show="!loading">
        <section
          class="login-bg"
          :class="[$vuetify.breakpoint.mdAndUp ? 'xl-full-width' : 'sm-full-width']"
        >
          <v-row class="justify-center-custom">
            <div
              class="cont mb-5"
              :class="[$vuetify.breakpoint.smAndUp ? 'small-width' : 'full-width']"
            >
              <div class="form-right">
                <div class="card-body">
                  <h3 class="text-center">
                    <!-- TODO: translate -->
                    <strong>Forgot Password?</strong>
                  </h3>
                  <form v-if="!message" @submit.prevent="submit()">
                    <p
                      class="text-center"
                      style="margin-top: 10px; margin-bottom: 10px"
                    >
                      Enter the email ID you used when you joined and we will send
                      you temporary password
                    </p>
                    <br />
                    <div class="form-group mb-50">
                      <label class="text-bold-600">{{ 'E-Mail Address' }}</label>
                      <input
                        id="email"
                        v-model="email"
                        type="email"
                        class="form-control"
                        :class="{ 'is-invalid': error.email }"
                        name="email"
                        required
                        autocomplete="off"
                        autofocus
                      />
                      <span
                        v-if="error.email"
                        class="invalid-feedback"
                        role="alert"
                      >
                        <strong>{{ error.email }}</strong>
                      </span>
                    </div>
                    <button
                      type="submit"
                      class="btn ml-0 btn-login btn-primary w-100"
                    >
                      <span
                        v-if="formSubmitting"
                        class="spinner-border spinner-border-sm mr-1"
                        role="status"
                        aria-hidden="true"
                      ></span>
                      {{ 'Send Password Reset Link' }}
                      <i class="fa fa-arrow-right"></i>
                    </button>
                  </form>
                  <div v-else>
                    <p
                      class="text-center text-success"
                      style="margin-top: 10px; margin-bottom: 10px"
                    >
                      {{ message }}
                    </p>
                  </div>
                </div>
              </div>
            </div>
          </v-row>
        </section>
      </div>
    </template>
    
    <script>
    import AssetLoader from '@/mixins/AssetLoader'
    
    export default {
      layout: 'landing',
      mixins: [AssetLoader],
      data() {
        return {
          error: {
            email: false,
          },
          email: null,
          loading: false,
          formSubmitting: false,
          message: '',
        }
      },
    
      async beforeDestroy() {
        await this.unloadCSS(
          'https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css'
        )
        await this.unloadCSS(
          'https://fonts.googleapis.com/css2?family=Poppins:ital,wght@0,100;0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,100;1,200;1,300;1,400;1,500;1,600;1,800;1,900&display=swap'
        )
        await this.unloadCSS('/css/mdb.min.css')
        await this.unloadCSS('/css/style.css')
        await this.unloadCSS('/css/landing-school.css')
        await this.unloadCSS('/css/landing-school-options.css')
        await this.unloadCSS(
          'https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css'
        )
        await this.unloadScript('/js/jquery-3.4.1.min.js')
        await this.unloadScript('/js/popper.min.js')
        await this.unloadScript('/js/bootstrap.min.js')
        await this.unloadScript('/js/popup.js')
        await this.unloadScript('/js/owl.carousel.js')
        await this.unloadScript('/js/jquery.nivo.slider.js')
        await this.unloadScript('/js/landing-school.js')
      },
      async mounted() {
        this.loading = true
        await this.loadCSS([
          'https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css',
          'https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css',
          'https://fonts.googleapis.com/css2?family=Poppins:ital,wght@0,100;0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,100;1,200;1,300;1,400;1,500;1,600;1,800;1,900&display=swap',
          '/css/mdb.min.css',
          '/css/landing-school.css',
          '/css/landing-school-options.css',
        ])
        await this.loadJS([
          '/js/jquery-3.4.1.min.js',
          '/js/popper.min.js',
          '/js/bootstrap.min.js',
          '/js/popup.js',
          '/js/owl.carousel.js',
          '/js/jquery.nivo.slider.js',
          // '/js/landing-school.js'
        ])
        this.loading = false
      },
    
      methods: {
        async submit() {
          this.formSubmitting = true
          this.error.email = ''
          try {
            const { data } = await this.$axios.post('/auth/forgot-password', {
              email: this.email,
            })
            this.message = data.status
            this.email = ''
            this.formSubmitting = false
          } catch (error) {
            this.formSubmitting = false
            if (error?.response?.data?.errorsArray?.length) {
              this.error.email = error?.response?.data?.errorsArray[0]
            } else if (error?.response?.data?.email) {
              this.error.email = error?.response?.data?.email
            } else {
              this.error.email = error.message
            }
          }
        },
      },
    }
    </script>
    
    <style scoped>
    .navbar {
      display: none !important;
    }
    #btn-amazon {
      background: #f90 !important;
      border-color: #f90 !important;
      color: #fff !important;
    }
    #btn-apple {
      background: #7e878b !important;
      border-color: #7e878b !important;
      color: #fff !important;
    }
    #btn-twitter {
      background: #32def4 !important;
      border-color: #32def4 !important;
      color: #fff !important;
    }
    .btn-login {
      background: -webkit-linear-gradient(45deg, #303f9f, #7b1fa2);
      background: linear-gradient(45deg, #303f9f, #7b1fa2);
      box-shadow: 3px 3px 20px 0 rgba(123, 31, 162, 0.5);
    }
    .form-left {
      padding: 114px 10px;
    }
    .cont {
      border-radius: 10px;
      display: block;
    }
    .cont.full-width {
      width: 90%;
    }
    .cont.small-width {
      width: 600px;
    }
    .row.justify-center-custom {
      justify-content: center;
    }
    .login-bg.xl-full-width {
      height: 100vh;
    }
    .login-bg.sm-full-width {
      height: 100%;
    }
    </style>

_token.vue

<template>
  <div v-show="!loading">
    <section
      class="login-bg"
      :class="[$vuetify.breakpoint.mdAndUp ? 'xl-full-width' : 'sm-full-width']"
    >
      <v-row class="justify-center-custom">
        <div
          class="cont mb-5"
          :class="[$vuetify.breakpoint.smAndUp ? 'small-width' : 'full-width']"
        >
          <div class="form-right">
            <div class="card-body">
              <h3 class="text-center">
                <!-- TODO: translate -->
                <strong>Reset Password</strong>
              </h3>
              <form v-if="!message" @submit.prevent="submit()">
                <p
                  class="text-center"
                  style="margin-top: 10px; margin-bottom: 10px"
                >
                  Enter the email ID you used when you joined and we will send
                  you temporary password
                </p>
                <br />
                <div v-if="errors.length" class="card px-3 py-3 mb-3">
                  <ol class="text-danger mb-0" style="list-style-type: none">
                    <li v-for="(error, index) in errors" :key="index">
                      <strong>{{ error }}</strong>
                    </li>
                  </ol>
                </div>
                <div class="form-group mb-50">
                  <label class="text-bold-600">{{ 'E-Mail Address' }}</label>
                  <input
                    v-model="email"
                    type="email"
                    class="form-control"
                    name="email"
                    autocomplete="off"
                    autofocus
                  />
                </div>
                <div class="form-group mb-50">
                  <label class="text-bold-600">{{ 'Password' }}</label>
                  <input
                    v-model="password"
                    type="password"
                    class="form-control"
                    autocomplete="off"
                    autofocus
                  />
                </div>
                <div class="form-group mb-50">
                  <label class="text-bold-600">{{ 'Confirm Password' }}</label>
                  <input
                    v-model="password_confirmation"
                    type="password"
                    class="form-control"
                    autocomplete="off"
                    autofocus
                  />
                </div>
                <button
                  type="submit"
                  class="btn ml-0 btn-login btn-primary w-100"
                >
                  <span
                    v-if="formSubmitting"
                    class="spinner-border spinner-border-sm mr-1"
                    role="status"
                    aria-hidden="true"
                  ></span>
                  {{ 'Reset Password' }}
                  <i class="fa fa-arrow-right"></i>
                </button>
              </form>
              <div v-else>
                <p
                  class="text-center text-success"
                  style="margin-top: 10px; margin-bottom: 10px"
                >
                  {{ message }}
                </p>
                <button
                  type="button"
                  class="btn ml-0 btn-login btn-primary w-100"
                  @click="$router.push('/login')"
                >
                  {{ $t('login_now') }}
                  <i class="fa fa-arrow-right"></i>
                </button>
              </div>
            </div>
          </div>
        </div>
      </v-row>
    </section>
  </div>
</template>

<script>
import AssetLoader from '@/mixins/AssetLoader'

export default {
  layout: 'landing',
  mixins: [AssetLoader],
  data() {
    return {
      errors: [],
      token: null,
      email: null,
      password: null,
      password_confirmation: null,
      loading: false,
      formSubmitting: false,
      message: '',
    }
  },

  created() {
    if (this.$route.params.token) {
      this.token = this.$route.params.token
    }
  },

  async beforeDestroy() {
    await this.unloadCSS(
      'https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css'
    )
    await this.unloadCSS(
      'https://fonts.googleapis.com/css2?family=Poppins:ital,wght@0,100;0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,100;1,200;1,300;1,400;1,500;1,600;1,800;1,900&display=swap'
    )
    await this.unloadCSS('/css/mdb.min.css')
    await this.unloadCSS('/css/style.css')
    await this.unloadCSS('/css/landing-school.css')
    await this.unloadCSS('/css/landing-school-options.css')
    await this.unloadCSS(
      'https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css'
    )
    await this.unloadScript('/js/jquery-3.4.1.min.js')
    await this.unloadScript('/js/popper.min.js')
    await this.unloadScript('/js/bootstrap.min.js')
    await this.unloadScript('/js/popup.js')
    await this.unloadScript('/js/owl.carousel.js')
    await this.unloadScript('/js/jquery.nivo.slider.js')
    await this.unloadScript('/js/landing-school.js')
  },
  async mounted() {
    this.loading = true
    await this.loadCSS([
      'https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css',
      'https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css',
      'https://fonts.googleapis.com/css2?family=Poppins:ital,wght@0,100;0,200;0,300;0,400;0,500;0,600;0,700;0,800;0,900;1,100;1,200;1,300;1,400;1,500;1,600;1,800;1,900&display=swap',
      '/css/mdb.min.css',
      '/css/landing-school.css',
      '/css/landing-school-options.css',
    ])
    await this.loadJS([
      '/js/jquery-3.4.1.min.js',
      '/js/popper.min.js',
      '/js/bootstrap.min.js',
      '/js/popup.js',
      '/js/owl.carousel.js',
      '/js/jquery.nivo.slider.js',
      // '/js/landing-school.js'
    ])
    this.loading = false
  },

  methods: {
    async submit() {
      this.formSubmitting = true
      this.errorsl = []
      try {
        const { data } = await this.$axios.post('/auth/reset-password', {
          email: this.email,
          token: this.token,
          password_confirmation: this.password_confirmation,
          password: this.password,
        })
        this.message = data.status
        this.email = ''
        this.formSubmitting = false
      } catch (error) {
        this.formSubmitting = false
        if (error?.response?.data?.errorsArray?.length) {
          this.errors = error?.response?.data?.errorsArray
        } else if (error?.response?.data?.email) {
          this.errors = [error?.response?.data?.email]
        }
      }
    },
  },
}
</script>

<style scoped>
.navbar {
  display: none !important;
}
#btn-amazon {
  background: #f90 !important;
  border-color: #f90 !important;
  color: #fff !important;
}
#btn-apple {
  background: #7e878b !important;
  border-color: #7e878b !important;
  color: #fff !important;
}
#btn-twitter {
  background: #32def4 !important;
  border-color: #32def4 !important;
  color: #fff !important;
}
.btn-login {
  background: -webkit-linear-gradient(45deg, #303f9f, #7b1fa2);
  background: linear-gradient(45deg, #303f9f, #7b1fa2);
  box-shadow: 3px 3px 20px 0 rgba(123, 31, 162, 0.5);
}
.form-left {
  padding: 114px 10px;
}
.cont {
  border-radius: 10px;
  display: block;
}
.cont.full-width {
  width: 90%;
}
.cont.small-width {
  width: 600px;
}
.row.justify-center-custom {
  justify-content: center;
}
.login-bg.xl-full-width {
  height: 100vh;
}
.login-bg.sm-full-width {
  height: 100%;
}
</style>

最终在 Plesk Apache 上完成此工作的答案是将 .htaccess 文件添加到与 index.html 相同的目录中。内容为:-

<IfModule mod_rewrite.c>
  RewriteEngine On
  RewriteBase /
  RewriteRule ^index\.html$ - [L]
  RewriteCond %{REQUEST_FILENAME} !-f
  RewriteCond %{REQUEST_FILENAME} !-d
  RewriteRule . /index.html [L]
</IfModule>

已解释here

Netlify 上的解决方案是为重定向构建添加一些特定配置。在正在部署的 repo 分支的根目录中创建 netlify.toml。

Netlify.toml 包含:-

[[redirects]]
  from = "/*"
  to = "/index.html"
  status = 200

指导自this

我将其解读为 - 如果 nuxt 生成了 html 文件,它将为其提供服务并且您的路线会很好。但是如果它没有生成与路由完全匹配的文件,那么您需要调用应用程序的入口点来初始化它并使路由可用。

不确定您的 Plesk + Nginx + Apache 特定配置,但将其托管在 Netlify 上仍然是最简单和最快的解决方案。

此外,如果可行,我会去 target: staticssr: true