Vue Axios 拦截器响应 Firebase 401 令牌 Expired/Refresh(未定义)
Vue Axios Interceptor Response Firebase 401 Token Expired/Refresh (undefined)
我在 Vuejs v2 网站中使用以下拦截器将 firebase 令牌推送到我的节点后端。在后端,我 detect/verify 令牌,使用 uid 从数据库中提取一些数据,然后处理任何 api 调用。
即使我使用 firebase onIdTokenChanged 来自动检索新的 ID 令牌,有时,如果用户已登录,但在一个小时内处于非活动状态,令牌会过期而不会刷新.现在,这没什么大不了的——我可以检查 axios 响应拦截器并将它们推送到登录页面,但这似乎很烦人,如果我可以检测到 401 令牌已过期,请重新发送 axios 调用并刷新令牌,如果用户碰巧与需要来自 API 调用的数据的组件交互,用户甚至不会知道它发生了。所以这就是我所拥有的:
main.js
Vue.prototype.$axios.interceptors.request.use(function (config) {
const token = store.getters.getSessionToken;
config.headers.Authorization = `Bearer ${token}`;
return config;
});
Vue.prototype.$axios.interceptors.response.use((response) => {
return response }, async function (error) {
let originalRequest = error.config
if (error.response.status === 401 && !originalRequest._retry) {
originalRequest._retry = true;
let user = auth.currentUser;
await store.dispatch("setUser", {user: user, refresh: true}).then(() => {
const token = store.getters.getSessionToken;
Vue.prototype.$axios.defaults.headers.common['Authorization'] = 'Bearer ' + token;
return Vue.prototype.$axios.request(originalRequest);
});
}
return Promise.reject(error); });
let app;
auth.onAuthStateChanged(async user => {
await store.dispatch("setUser", {user: user, refresh: false}).then(() => {
if (!app) {
app = new Vue({
router,
store,
vuetify,
render: h => h(App)
}).$mount('#app')
}
})
.catch(error => {
console.log(error);
});
});
vuex
setUser({dispatch, commit}, {user, refresh}) {
return new Promise((resolve) => {
if(user)
{
user.getIdToken(refresh).then(token => {
commit('SET_SESSION_TOKEN', token);
this._vm.$axios.get('/api/user/session').then((response) => {
if(response.status === 200) {
commit('SET_SESSION_USER', response.data);
resolve(response);
}
})
.catch(error => {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: 'Server unavailable: '+error
});
resolve();
});
})
.catch(error => {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: 'Unable to verify auth token.'+error
});
resolve();
});
}
else
{
console.log('running logout');
commit('SET_SESSION_USER', null);
commit('SET_SESSION_TOKEN', null);
resolve();
}
})
},
我在 vuex 中设置令牌,然后在所有 API 调用的拦截器中使用它。所以我在这段代码中看到的问题是,我正在使用过期令牌向后端发出 API 调用。这个 returns 一个 401 和 axios 响应拦截器将其拾取并完成刷新 firebase 令牌的过程。然后,这会使用与更新后的令牌相同的配置向后端发出新的 API 调用,并 returns 向原始 API 调用(如下)。
这一切 似乎 工作,我可以在开发 tools/network 中看到,API 调用的响应正在发回正确的数据。不过,好像落入下面apicall/code的圈套了。例如,当我尝试使用 response.data.server 加载表单字段时,我得到一个 "undefined"。如果我刷新页面(同样,正常 token/loading 过程应该如此),此页面会正常加载所有内容,所以我知道没有加载问题。
vue组件(加载smtp设置到页面)
getSMTPSettings: async function() {
await this.$axios.get('/api/smtp')
.then((response) => {
this.form.server = response.data.server;
this.form.port = response.data.port;
this.form.authemail = response.data.authemail;
this.form.authpassword = response.data.authpassword;
this.form.sendemail = response.data.sendemail;
this.form.testemail = response.data.testemail;
this.form.protocol = response.data.protocol;
})
.catch(error => {
console.log(error);
});
},
我已经看了几天了,我不明白为什么它不会加载它。数据似乎在那里。我正在做的事情的时机是否导致了我的问题?它似乎不是 CORS 问题,我在那里没有收到任何错误。
您的主要问题是将异步/等待与 .then()
混用。您的响应拦截器没有返回下一个响应,因为您已经将该部分包装在 then()
中而没有返回外部承诺。
使用 async / await 让事情变得简单。
此外,设置 common headers 会破坏使用拦截器的意义。你已经有了一个请求拦截器,让它完成它的工作
// wait for this to complete
await store.dispatch("setUser", { user, refresh: true })
// your token is now in the store and can be used by the request interceptor
// re-run the original request
return Vue.prototype.$axios.request(originalRequest)
你的店铺操作也属于explicit promise construction antipattern,可以简化
async setUser({ dispatch, commit }, { user, refresh }) {
if(user) {
try {
const token = await user.getIdToken(refresh);
commit('SET_SESSION_TOKEN', token);
try {
const { data } = await this._vm.$axios.get('/api/user/session');
commit('SET_SESSION_USER', data);
} catch (err) {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: `Server unavailable: ${err.response?.data ?? err.message}`
})
}
} catch (err) {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: `Unable to verify auth token. ${error}`
})
}
} else {
console.log('running logout');
commit('SET_SESSION_USER', null);
commit('SET_SESSION_TOKEN', null);
}
}
我在 Vuejs v2 网站中使用以下拦截器将 firebase 令牌推送到我的节点后端。在后端,我 detect/verify 令牌,使用 uid 从数据库中提取一些数据,然后处理任何 api 调用。
即使我使用 firebase onIdTokenChanged 来自动检索新的 ID 令牌,有时,如果用户已登录,但在一个小时内处于非活动状态,令牌会过期而不会刷新.现在,这没什么大不了的——我可以检查 axios 响应拦截器并将它们推送到登录页面,但这似乎很烦人,如果我可以检测到 401 令牌已过期,请重新发送 axios 调用并刷新令牌,如果用户碰巧与需要来自 API 调用的数据的组件交互,用户甚至不会知道它发生了。所以这就是我所拥有的:
main.js
Vue.prototype.$axios.interceptors.request.use(function (config) {
const token = store.getters.getSessionToken;
config.headers.Authorization = `Bearer ${token}`;
return config;
});
Vue.prototype.$axios.interceptors.response.use((response) => {
return response }, async function (error) {
let originalRequest = error.config
if (error.response.status === 401 && !originalRequest._retry) {
originalRequest._retry = true;
let user = auth.currentUser;
await store.dispatch("setUser", {user: user, refresh: true}).then(() => {
const token = store.getters.getSessionToken;
Vue.prototype.$axios.defaults.headers.common['Authorization'] = 'Bearer ' + token;
return Vue.prototype.$axios.request(originalRequest);
});
}
return Promise.reject(error); });
let app;
auth.onAuthStateChanged(async user => {
await store.dispatch("setUser", {user: user, refresh: false}).then(() => {
if (!app) {
app = new Vue({
router,
store,
vuetify,
render: h => h(App)
}).$mount('#app')
}
})
.catch(error => {
console.log(error);
});
});
vuex
setUser({dispatch, commit}, {user, refresh}) {
return new Promise((resolve) => {
if(user)
{
user.getIdToken(refresh).then(token => {
commit('SET_SESSION_TOKEN', token);
this._vm.$axios.get('/api/user/session').then((response) => {
if(response.status === 200) {
commit('SET_SESSION_USER', response.data);
resolve(response);
}
})
.catch(error => {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: 'Server unavailable: '+error
});
resolve();
});
})
.catch(error => {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: 'Unable to verify auth token.'+error
});
resolve();
});
}
else
{
console.log('running logout');
commit('SET_SESSION_USER', null);
commit('SET_SESSION_TOKEN', null);
resolve();
}
})
},
我在 vuex 中设置令牌,然后在所有 API 调用的拦截器中使用它。所以我在这段代码中看到的问题是,我正在使用过期令牌向后端发出 API 调用。这个 returns 一个 401 和 axios 响应拦截器将其拾取并完成刷新 firebase 令牌的过程。然后,这会使用与更新后的令牌相同的配置向后端发出新的 API 调用,并 returns 向原始 API 调用(如下)。
这一切 似乎 工作,我可以在开发 tools/network 中看到,API 调用的响应正在发回正确的数据。不过,好像落入下面apicall/code的圈套了。例如,当我尝试使用 response.data.server 加载表单字段时,我得到一个 "undefined"。如果我刷新页面(同样,正常 token/loading 过程应该如此),此页面会正常加载所有内容,所以我知道没有加载问题。
vue组件(加载smtp设置到页面)
getSMTPSettings: async function() {
await this.$axios.get('/api/smtp')
.then((response) => {
this.form.server = response.data.server;
this.form.port = response.data.port;
this.form.authemail = response.data.authemail;
this.form.authpassword = response.data.authpassword;
this.form.sendemail = response.data.sendemail;
this.form.testemail = response.data.testemail;
this.form.protocol = response.data.protocol;
})
.catch(error => {
console.log(error);
});
},
我已经看了几天了,我不明白为什么它不会加载它。数据似乎在那里。我正在做的事情的时机是否导致了我的问题?它似乎不是 CORS 问题,我在那里没有收到任何错误。
您的主要问题是将异步/等待与 .then()
混用。您的响应拦截器没有返回下一个响应,因为您已经将该部分包装在 then()
中而没有返回外部承诺。
使用 async / await 让事情变得简单。
此外,设置 common headers 会破坏使用拦截器的意义。你已经有了一个请求拦截器,让它完成它的工作
// wait for this to complete
await store.dispatch("setUser", { user, refresh: true })
// your token is now in the store and can be used by the request interceptor
// re-run the original request
return Vue.prototype.$axios.request(originalRequest)
你的店铺操作也属于explicit promise construction antipattern,可以简化
async setUser({ dispatch, commit }, { user, refresh }) {
if(user) {
try {
const token = await user.getIdToken(refresh);
commit('SET_SESSION_TOKEN', token);
try {
const { data } = await this._vm.$axios.get('/api/user/session');
commit('SET_SESSION_USER', data);
} catch (err) {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: `Server unavailable: ${err.response?.data ?? err.message}`
})
}
} catch (err) {
dispatch('logout');
dispatch('setSnackbar', {
color: "error",
timeout: 4000,
text: `Unable to verify auth token. ${error}`
})
}
} else {
console.log('running logout');
commit('SET_SESSION_USER', null);
commit('SET_SESSION_TOKEN', null);
}
}