-
-
Notifications
You must be signed in to change notification settings - Fork 649
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Migrate react-scripts to vite #696
base: master
Are you sure you want to change the base?
Conversation
Ok, to sum it up. I tried to update the router, but that is even worse. They broke the complete API and do not even provide a migration script for it. As I mainly develop in Angular I need to read about react before I continue, but it seems that this upgrade to vite is depending on updating all libs to a more current version. If someone else here have a strategy how to update the ui, any feedback is welcome. |
@mfechner I feel you (:, I think the only good way it to start from scratch and copy component by component into a new project, so that you don't have to fix 100 errors to get something working. It's on my todo list, sadly it's multi hour effort. |
@jmattheis thanks a lot for your feedback. I currently study react to better understand react itself. I'm on this task but it will take some time ;) |
The codebase is quite old, pre react hooks. In my experience it wasn't that ergonomic to use contexts for global state, so I've used a lib. Currently I like https://github.com/pmndrs/zustand for it's simplicity and small footprint. |
I started now with a rewrite, but this will be a massiv change.
I think it will be nearly impossible to get this changes sliced into small commit, as all steps link somehow together. I update you, if I have something to show, but it will for sure take some days maybe some weeks. |
f296d85
to
1bf4902
Compare
In DEV mode, you just do: go run . (cd ui && yarn start) In dev mode there is a proxy in vite used to not have CORS problem. You can access the UI on: http://localhost:5173 The UI prefix every request to the backup with `/api` so the proxy can correctly handle it, the proxy strips the `/api` so the backend is equal to production mode.
changing (observed) observable values without using an action
1bf4902
to
3d6fca9
Compare
This is now a first version we can start. There are some todos left (infinite loading component for messages), but it is now a good starting point. The only problem I have is that puppeteer sometimes is just not loading the page and therefore tests are not running. @jmattheis would be nice if you can have a look, it was really a lot of work |
All tests are now working fine for me. |
I'll have a look at this in the coming days, but I'll take some time. |
thank you very much. I found other small issues I corrected in the meantime. The problems I found is that for most actions a submit or a click on a button was not blocking (running in background), which I reverted to the behaviour the released gotify version has. Later we can addnice indicators, if you e.g. click on Submit the button is replaced by an loading indicator. I have here some very nice ideas, but I will provide this later in another PR, as that is not related to this topic. Let us first get the update to react 19 merged, then we can continue further in smaller steps. Maybe we can release these changes as a beta version to enable users testing it. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not finished with reviewing, this was only glancing through some files.
} | ||
|
||
const initialUiState: UiState = { | ||
themeKey: 'dark', |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure I like this too, previously this was local component state, because it was only used in two places, and passed downwards to one component. Sure adding this to the global state can be done, but local component state is better as it's simpler and easier to navigate.
These refactorings making this review much more effort as it's not just library change / upgrades, but also logic changes which aren't 100% necessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I had a lot of problems in the beginning to understand which prop is passed to which component and why
Using a global state reduces the need of props drilling which is making the componets match leaner and easier to read.
But if you prefer a global state in e.g. app.tsx, I can remove these UI related states from the global state and move it to app.tsx and pass them via props drilling to the components.
But to be honest it will be technically equal and will just make the components harder to read.
But as you prefer.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I prefer it to be as close as possible to the original implementation for now, as it makes less changes to review and less possibility of bugs, because the code already runs for some time. This includes the moving of the selected plugin state into redux.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ok, I will move the state/slice/action definition to the component.
But I will not remove the ui state management and will change everything back to prop drilling.
This is error prone and we will for sure just add new bugs by this. The UI state is working perfectly fine.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is error prone and we will for sure just add new bugs by this.
How is it error prone to pass properties to sub components? TypeScript is used, so it's not like you can miss passing the props?
Rewrite all components to functional component using hooks Upgrade react-router to 7.1.0 Use reduxjs/toolkit and react-redux for state management Upgrade material to 6.3.0 Updated most dependencies to current versions
This caused a failure correctly forwarding the user to the login page
…o missing network
Replace react-codemirror2 which is unmaintained by @uiw/react-codemirror and applied required API changes to ReactMarkdown
… the store more than once
…container and can be accessed for the windows host. This is required to use/test plugins with gotify
while working with the production bundled application.
…ld not happen, as the key on already rendered table does not change)
We hold now all messages in the state. If message of one app are refresh, make sure to sort them by date into the existing messages
Axios does not need this for error handling
…f the matching component
Except for the delete actions, as the test does not expect it there
92a9608
to
4d0f9bf
Compare
I removed now some commits and placed them into different branches.
They belong in a way to the migration (to restore the original functionality), but can be merged within another PRs. Paging of message (if it is ok for you), I will implement later, so we do not have to many branches we need to merge in correct order. |
Depending on my decision in #696 (comment) I'd want the same functionality as on current master in this PR. There is another open discussion which is unanswered. |
"@uiw/react-codemirror": "^4.23.7", | ||
"axios": "1.7.9", | ||
"detect-browser": "5.3.0", | ||
"install": "^0.13.0", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this dep used somewhere?
"axios": "1.7.9", | ||
"detect-browser": "5.3.0", | ||
"install": "^0.13.0", | ||
"js-base64": "3.7.7", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This dep shouldn't be needed, base64 encoding/decoding can be done natively.
"react-router": "7.1.1", | ||
"react-router-dom": "7.1.1", | ||
"react-timeago": "7.2.0", | ||
"react-window": "1.8.11", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
unused
reaction( | ||
() => stores.currentUser.connectionErrorMessage, | ||
(connectionErrorMessage) => { | ||
if (!connectionErrorMessage) { | ||
clearAll(); | ||
loadAll(); | ||
} | ||
} | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems to be not migrated into redux.
const loggedIn = useAppSelector((state) => state.auth.loggedIn); | ||
|
||
useEffect(() => { | ||
if (loggedIn) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this really the way to react to changes in the state, it there a native way in redux to subscribe to changes in the state and do stuff? Doing this in react seems weird, and having the global state besides redux store (the websocket store varibale in this case, or reconnectTimeoutId in the auth-actions.ts) seems weird to. The redux actions shouldn't have side effects with other global variables.
const [createDialog, setCreateDialog] = useState<boolean>(false); | ||
|
||
const fileInputRef = useRef<HTMLInputElement>(null); | ||
let uploadId = -1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not sure why this works, but this should probably be a React.useRef(), otherwise it will be overriden by a rerender.
.catch(() => Promise.resolve()); | ||
|
||
localStorage.removeItem(tokenKey); | ||
dispatch(authActions.logout()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This naming is weird. In a file called auth-actions, we have a function called logout which uses authActions.logout from auth-slice.ts. Is this really the redux way? Having a filed called as a variable from another file which use each other seems complicated.
Fixes #695
To test this modification:
Now access the address
yarn start
shows you.