-
-
Notifications
You must be signed in to change notification settings - Fork 115
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
[Request Interceptor] Refresh token and update global option #268
Comments
Here is a cleaned up version below, but I need some runnable code in a codesandbox import useLocalStorage from 'react-use-localstorage'
const FetchProvider = props => {
const [token, setToken] = useLocalStorage('token') // try using this package
const [refresh, setRefresh] = useLocalStorage('refresh')
const url = 'http://127.0.0.1:8000'
const [request, response] = useFetch(url);
const getToken = async () => {
console.log("is_executed")
let bodyFormData = new FormData();
bodyFormData.append('refresh', refresh);
let newToken = await request.post('/api/user/token_jwt/refresh/', bodyFormData)
if (response.ok) return newToken.access
console.error('there was a problem authenticating the user')
}
const globalOptions = {
interceptors: {
async request({ options }) { // <- probably need to add curly braces here
options.headers = {
...(options.headers || {}),
Authorization: `Bearer ${token}`
}
console.log("OPTION_ENTREE=", options)
if (isExpired(token)) {
const new_token = await getToken()
options.headers.Authorization = `Bearer ${new_token}`
setToken(new_token)
}
return options
}
}
}
if (isExpired(refresh)) return <Redirect to="/#" />
return <Provider {...props} url={url} options={globalOptions} />
}
export default FetchProvider and export default function DietHistory(props) {
const [request, response] = useFetch(); // no need for 2 of these
const [recipes, updateRecipes] = useState([]);
const [sent_recipes, updateSentRecipes] = useState([]);
const [spinner_on_off, setSpinner] = useState(false);
const [open_snackbar, setSnackBar] = useState(false);
const [recipe_being_sent, setRecipe_being_sent] = useState([
"owner_email",
0,
]);
const { state: authState } = useContext(AuthContext);
useEffect(() => {
initializeRecipes();
}, []);
async function initializeRecipes() {
const initialRecipes = await request.get("/api/recipe/recipe_list/");
if (response.ok) updateRecipes(initialRecipes);
}
async function sendPDF(recipe_id, recipe_email) {
setRecipe_being_sent([recipe_email, recipe_id]);
setSpinner(true);
setSnackBar(true);
const send_pdf = await request.get(
"/api/recipe/send-email?id=" + recipe_id
);
if (request.ok) updateSentRecipes([...sent_recipes, recipe_id]);
setSpinner(false);
setTimeout(() => {
setSnackBar(false);
}, 2000);
}
if (request.loading) return <CircularProgress />;
if (request.error) return <p>Error!</p>;
if (recipes) {
return (
<div>
<div style={{ maxWidth: "100%" }}>
<MaterialTable
icons={tableIcons}
columns={[
{
title: "Print PDF",
field: "PDF",
render: (rowData) => {
return <ButtonPrintPdf recipe_id={rowData.id} />;
},
},
{
title: "Send PDF",
field: "PDF",
render: (rowData) => {
return (
<ButtonLoadding
email={rowData.owner_email}
loading={spinner_on_off}
sent_recipes={sent_recipes}
recipeid={rowData.id}
recipeid_being_sent={recipe_being_sent[1]}
tool_to_sendPDF={sendPDF}
/>
);
},
},
]}
data={recipes}
/>
</div>
</div>
);
}
} |
|
start by forking this codesandbox and try to reproduce. Then put a link to your reproduced bug here. |
Hey @alex-cory, Behaviour :
That's the closest I'm getting from the real bug. The real bug I'm just getting the old expired token. What's your take on this ? |
@lgm-dev Hi, I think I have the same problem with you, and here is my sandbox example , it may be more simple than yours. I found the Provider can not read the real time state value, it may cache some thing. Hi, @alex-cory , Am I missing something ? |
I will take a look tomorrow. Heading to sleep. It's 2:13am. |
I think I know what happened... I don't know if it's a bug... @alex-cory Here is the reason: https://github.com/ava/use-http/blob/master/src/Provider.tsx#L14 Because you used For now, we can do this for avoid that : const App = () => {
const [token, setToken] = useState()
useEffect(() => {
setTimeout(() => {
setToken("new token")
}, 1000)
}, [])
const globalOptions = {
headers:{
token:token,
},
interceptors: {
request: ({ options }) => {
options.headers = {
Authorization: `Bearer ${token}`
}
console.log("interceptors token", token)
return options
},
response: ({ response }) => {
console.log("initial resopnse.data", response.data)
return response
}
}
}
return (
<Provider options={globalOptions}>
<TestUseFetch token={token} />
</Provider>
)
} |
Thanks for the idea @theowenyoung , but I fear that if we proceed this way, the point of intercepting the request sounds a bit useless. |
This issue is similar (or likely the same) as mine, and I've been hesitating to suggest this, but I've tried using a different means of determining deep equality on the following line and it seems to fix the issue: Line 235 in 0656ec5
At the risk of coming across as ignorant, I believe the reason this works is because the interceptor references change here: Line 36 in 0656ec5
...and while a deep equality function can pick up on the changes to function references, JSON.stringify cannot. The options are also memoized, and the interceptor function references should change each time the options change. This should happen when the provider gets re-rendered, so the function references shouldn't change an indefinite number of times. Comparing function references might cause other issues outside of interceptors, but since the interceptors are already memoized, and I don't think there are any other functions in the dependency array of makeFetch (as far as I can see) that aren't memoized in some way, it should work. |
Hey Guys, |
Apologies guys. I've been dealing with a lot of personal/family issues recently. I will get to this asap. |
@theowenyoung : give a desperate try to your solution today. No success (again). Even when you pass the token down to the children component, the @CaveSeal : in another issue you seemed to have found a simpler solution to this token-refreshing problem ? anything to help :) ? @alex-cory : good luck with your issues :'( |
@lgm-dev The way I solved my issue won't work for you, because I didn't need to do any checks (i.e. for expiry) on each request. However, the issue of being unable to see an updated token in the interceptor is (I believe) due to the object comparison that happens when the various options are memoized, which doesn't take account for changes in function references. The function references never change, so you always see the same outer environment, which means you'll only see the token as it was initially. Wouldn't call myself a Javascript expert though, so this is all wild speculation. One thing that might work as a workaround (though not ideal) is if you retrieved your token from local storage inside the interceptor. It's also likely that you'll have to do the same with the refresh token.
I've got a PR up that I believe solves this issue and all similar issues, but we'll see what happens. |
I'm facing this issue too, any update about when this will be solved/published? I can help with a PR or anything to delivery this bug fix, let me know how I can help |
I am having the same issue right know and I was wondering, if there is any solution to this. |
Hi,
I have been struggling for two days on this. I'm not sure if it's a bug or me not getting the art of this Provider feature.
I have this Provider :
The child component is as follow :
So when I hit
sendPDF
after the token has expired, the provider doesn't seem to update the globalOption and just send back the previous token (which has now expired), causing a401 error
.However If I just re-render the whole children component, then the token is nicely refreshed and pass to all of my global option.
It seems like global option cannot be updated on the go (conditionally, when the token has expired for exemple).
Am I missing something ?
The text was updated successfully, but these errors were encountered: