Skip to content

⏯ A toy GitHub data explorer which I'm using to explore frontend tech like GraphiQL, React hooks, and Redux.

Notifications You must be signed in to change notification settings

dgroomes/my-github-explorer

Repository files navigation

my-github-explorer

NOT YET FULLY IMPLEMENTED

⏯️ A toy GitHub data explorer which I'm using to explore frontend tech like GraphiQL, Redux, React, TypeScript and Electron.

Overview

This is a mash-up of front-end tech I'm interested in learning more deeply by way of a realistic application. I often learn libraries, languages and frameworks by way of minimal runnable examples, but these small examples don't let complex integration scenarios emerge. This project is an SPA that uses the GitHub GraphQL API and GraphiQL components to fetch metadata about repositories and present it using React. It also uses Redux and Electron. The amount of side-effecting, stateful code and UI complexity is enough to really showcase how this tech works.

This is not an endorsement for any of these technologies.

Instructions

Follow these instructions to build and serve the application:

  1. Pre-requisite: Node.js
    • I used the version of Node specified in the .nvmrc file. If you are using nvm then you can run the following command to use the correct version of Node.
    • nvm use
  2. Pre-requisite: build-support
    • Important: The build-support library must be built before you can develop the main application. Follow the instructions in the build-support README. You only need to do this the first time and then any time you change build-support. Re-install it with the following command.
    • npm install --save-dev ./build-support/my-github-explorer_build-support-1.0.0.tgz
  3. Install dependencies
    • npm install
  4. Continuously build and run the app
    • npm start
  5. Make the app distribution
    • npm run make
    • You will then need to find the .dmg file in the out directory and install it.

Tip: the GitHub personal access token is saved for convenience in an unencrypted file in a conventional directory in the Electron app's installation directory. But you might want to blow this away to force the app through another flow. Use this alias:

alias my-github-explorer-reset-token='rm "$HOME/Library/Application Support/my-github-explorer/pat"'

Instructions for React and Redux Developer Tools

When you develop a React or Redux application, you'll likely want the power of their excellent developer tools: React Developer Tools and Redux DevTools. In fact, using React and Redux come at a price and these tools are essential to offset the cost.

Because we're in an Electron environment, it's a little tricky to install browser extensions, and it's an especially tricky situation in 2024 with the inconsistent support for Manifest v2 and v3. Electron doesn't support Manifest v3 yet but React Developer Tools doesn't support v2 anymore. Tricky. Thankfully, React Developer Tools and Redux DevTools can each be run in a standalone mode by way of their npm-distributed CLI programs:

My preference is to run these standalone tools. By contrast, I am interested in using browser extensions in Electron, but I'm not ready to figure out the compatibility workarounds (aforementioned v2/v3 quagmire).

Follows these instructions to install and run the developer tools and connect to them from our app:

  1. Install the tools globally:
    • npm install -g react-devtools @redux-devtools/cli@3.0.2
  2. Run React Developer Tools in standalone mode:
    • react-devtools
  3. Run a Redux DevTools server, and establish a connection from the web app
    • redux-devtools --open=browser --port=8000 --hostname=127.0.0.1
    • This starts the server and opens your web browser to http://127.0.0.1:8000. You need to configure the web app to connect to the server by going to the "Settings" tab and then clicking the "Use local (custom) server" radio button and clicking the "Connect" button. This is a one-time setup because this setting is stored in local storage.
  4. Run our app but with a special flag to connect to the standalone React Developer Tools:
    • npm run start:with-devtools
  5. Inspect the React components and Redux state.
    • Go back to the web app that hosts the Redux DevTools UI and refresh the page (I don't get why this is necessary).
    • You'll see the Redux state and state history of the program.
    • Go to the React Developer Tools Electron window and inspect the React components.
    • Develop -> Debug -> Develop -> Debug etc...

Wish List

General clean-ups, todos and things I wish to implement for this project:

  • Query (search) for the current user's repositories. Hardcode to a 100 limit (which is a limit of the API; then you'll need pagination)
  • Query (get) the metadata for each of the repositories
  • Display the information in a table using React Table
  • Consider paginating through the results. This would be a cool application because we would see the table size grow.
  • Show the pre-constructed queries in codemirror (using a GraphiQL component). We want the syntax highlighting. Can we make them read-only?
  • Consider enforcing noImplicitAny
  • DONE (I'm struggling with the program design; using too much globals) Move more code from useEffect to Redux listener callbacks/functions.
    • Consider removing useToken entirely. I'm acutely aware of the "fetch on render" (or rather, "side-effect on render") program construction. While it's often fine, it's a conflation I think. In these context, David Khourshid's advice really rings true about saying goodbye to useEffect.
    • After pushing logic into token.ts, revisit the exported components. We can probably encapsulate more.
  • RTK 2.0 ready? RTK 2.0 is a big upgrade, and I'm eager to upgrade. But Redux Developer Tools are not compatible with RTK 2.0 yet. Only upgrade when it's compatible. See more info in my own extensive notes.
  • Where am I supposed to put Redux-coupled code? In store.ts, -slice.ts, or somewhere else?
  • DONE I'm bothered by modules that do top-level work. Does the store.ts file really need to be the way it is? Why do we use globals so brazenly?
    • DONE Consider combining more token stuff into the same file. I'm just experimenting here.
  • Bring back useAppDispatch now that I understand utility types better.
  • DONE Rename app.tsx to index.tsx. I think I like the naming convention of "index" for web pages and I guess in an Electron context the word "main" is a good convention for the main process.

Finished Wish List Items

  • DONE Scaffold the project
  • DONE Prompt for an access token. Validate the access token with a request to the GitHub API. Give informational UI feedback.
  • DONE Query for the current user's login (assuming the token is a user token)
  • DONE (I like it very much) Model the PAT using some kind of types. I want to model the state machine a lot more cleanly than the mess of useState and nullability I have now.
  • DONE (I accomplished an abstraction, but it is not great) Abstract the "useEffect unmount clean up fetch abort" stuff into generic code. I'm assuming there must be some concise API to derive for this common use case.
  • DONE Abstract the token validation into a hook
  • DONE Extract into its own repository
  • DONE Turn this into an Electron app (primarily because I want native secrets)
    • This was extremely larger than I anticipated. Went down a rabbit hole here: https://github.com/dgroomes/electron-playground/tree/main/realistic I'll port over the same thing.
    • DONE Port over build-support
    • DONE Port over package-json.mjs style.
    • DONE Add main.js file for Electron main process.
    • DONE .gitignore
    • DONE Get it to build and add instructions to README.
    • DONE tsconfig changes
    • DONE Get the styles to work. I need to figure out how to import the CSS from GraphiQL again.
    • Make sure React DevTools works
  • DONE Upgrade dependencies.
  • DONE (GraphiQL defines fonts in data URLs) Why are there HTTP request failures to download fonts? E.g. data:font/woff2;base64, etc. This happens when serving but not in the production app.
  • DONE Get token from storage using Electron's safeStorage API (well safe is always relative). This involves code across the main and renderer processes. Note: I've been writing hard-to-reason-about state-checking code to infer what work my code needs to take next. This is awkward because we're reconstituting the transition from the state but what would be more natural is if we could just do the work (or queue it up?) at the time we made the state change (i.e. transition) in the first place. For example, I set state to "storing" but then at a later invocation of the render function I use a combination of checking for "storing" and also useEffects dependencies to actually kick off the storing work. But it's like... at this point we're actually doing the storing work. Should I model it as "need-to-store" and "storing" and just jam descriptive states in the state? I don't really get it, but the nature of an async domain is tricky regardless.
  • DONE Use Redux. I think I want to use Redux Sagas specifically. I don't grok Redux. I need to learn it an app that "does real things" like make HTTP requests and handles a rich UI. Hopefully I'll get it. I'm expecting the Redux devtools will be particularly useful, but jury is out.
    • DONE Start by bringing in the dependencies and maybe doing a "hello world".
    • DONE Port one instance of setState to Redux. The lowest level (the leaf node) is where useEffect uses setState. We'll port over all other usages and the useEffect usages in a later task.
      • DONE Redux wants only serializable data in the state. Makes sense (although profoundly limiting; but constraints can be good). Take only what I need in place of the Response object.
    • DONE Port everything else to Redux (but don't get into Sagas yet)
  • DONE Add Redux DevTools.
    • Note: I completely misread the docs. I thought there was no standalone launcher for Redux DevTools, but there is. It's called redux-devtools-cil.
    • DONE Add instructions for running Redux DevTools from its CLI.
    • DONE Add Redux DevTools client side stuff and make sure it can be connected to from the dev tool instance.
    • DONE Co-opt the start:react-devtools script to also start the Redux DevTools. And figure out how to exclude the package from the production build.
  • DONE Consider adding Prettier or something. I'm mostly annoyed with arbitrarily using double and single quotes and using and not using semicolons.
  • DONE (fixed, but useFetch now doesn't make sense. How do people do this? Just ignore unmounts for the clean up function?) Defect: validation fetch request is getting cancelled prematurely. My useFetch must be buggy.
  • DONE Stop setting state from render function. During the redux conversion, I started getting warnings about setting state from the render function. Totally (well 80%) makes sense to me, so I'll fix it. This is part of the process of grokking React. The trouble is in useToken. At this time, it's time to drop useFetch which I had previously marked as deprecated. It's so hard to make this work.
  • DONE More robust error handling. We at least want to model basic error states and propagate a basic message.
  • SKIP (I get the gist of this technology/paradigm but I don't see how I can be succesful/effective with it. Maybe on another project) Port to Redux Sagas
    • This is the real experiment: can we actually figure out how to implement the logic in Sagas?
  • DONE (I used a store listener to solve this; happy with it and what I learned about Redux and RTK) (I need to figure out how to do async stuff using Redux I think. So Sagas or thunk?) Defect. The validation request is doubling. I mean I imagine this is because of React strict mode. How am should I deal with this?
    • Side note: I've spent about 15 minutes researching React strict mode and I can't find in the official docs, stackoverflow or the source code what is meant by "strict mode only runs in development mode". Did the whole world agree that "dev web pages" have a global variable named __DEV__ (or something like that) set to true?
    • I need to not tie the fetch lifecycle to the UI component (I guess this is kind of obvious). I just want to spell it out for myself. Letting the mounting of a component trigger useEffect is fine in some cases, but when there is work/state that exists even "before" or "outside" a UI component then manage that in non-UI code (and to me the "render" function, which is a React function component is a UI thing). This is continually a murky subject for me (and most). OR actually, it's just silly to initialize the state to "restoring"... I should really do "init" or "untriggered" (I keep revisiting this same concept...). UPDATE: I've heard the phrase "fetch on render" used in the wild and that's a good description.
    • UPDATE: RTK has createListenerMiddleware. That's what I'll use.
    • UPDATE: Ok I am going to use a "kick-off token restoration on render" strategy. I'm using Redux listeners and state so that React's strict mode double effect-triggering is countered (phew, feels good).
  • SKIP (I don't want to do this. "Components encapsulating long-lived state" is wrong. Components are flighty because of the nature of React. I'm happy with my token state/logic in Redux, or even organic JS) Abstract the token input/storage into a component
  • OBSOLETE (I think this is not true. I think I am getting all the state. Refreshing the page I think is the important part) Figure out some trick to prevent the app from initializing until Redux DevTools have attached. I don't think I'm getting all the early action history which is a shame. If I can get a comprehensive history, then I can delete some of the log statements.

Reference

About

⏯ A toy GitHub data explorer which I'm using to explore frontend tech like GraphiQL, React hooks, and Redux.

Topics

Resources

Stars

Watchers

Forks