Skip to content

Commit 52700da

Browse files
Merge pull request #36 from developmentseed/feature/deployment
Fix deployment related issues
2 parents 75469ec + 0e5236c commit 52700da

File tree

8 files changed

+172
-19
lines changed

8 files changed

+172
-19
lines changed

.github/workflows/deploy-gh.yml

+10-10
Original file line numberDiff line numberDiff line change
@@ -26,27 +26,30 @@ jobs:
2626
uses: actions/checkout@v4
2727

2828
- name: Use Node.js
29-
uses: actions/setup-node@v3
29+
uses: actions/setup-node@v4
3030
with:
3131
node-version-file: '.nvmrc'
3232

3333
- name: Cache node_modules
34-
uses: actions/cache@v3
34+
uses: actions/cache@v4
3535
id: cache-node-modules
3636
with:
3737
path: node_modules
38-
key: ${{ runner.os }}-build-${{ env.cache-name }}-${{ hashFiles('**/package.json') }}
38+
key: ${{ runner.os }}-build-${{ hashFiles('**/package.json') }}
3939

4040
- name: Cache dist
41-
uses: actions/cache@v3
41+
uses: actions/cache@v4
4242
id: cache-dist
4343
with:
4444
path: packages/client/dist
45-
key: ${{ runner.os }}-build-${{ env.cache-name }}-${{ github.sha }}
45+
key: ${{ runner.os }}-build-${{ github.sha }}
4646

4747
- name: Install
4848
run: npm install
4949

50+
- name: Setup SPA on Github Pages
51+
run: node packages/client/tasks/setup-gh-pages.mjs
52+
5053
- name: Build
5154
run: npm run all:build
5255

@@ -59,14 +62,11 @@ jobs:
5962
uses: actions/checkout@v4
6063

6164
- name: Restore dist cache
62-
uses: actions/cache@v3
65+
uses: actions/cache@v4
6366
id: cache-dist
6467
with:
6568
path: packages/client/dist
66-
key: ${{ runner.os }}-build-${{ env.cache-name }}-${{ github.sha }}
67-
68-
- name: Copy index as 400 file for github pages
69-
run: cp packages/client/dist/index.html packages/client/dist/400.html
69+
key: ${{ runner.os }}-build-${{ github.sha }}
7070

7171
- name: Deploy 🚀
7272
uses: JamesIves/github-pages-deploy-action@v4

packages/client/.env

+4-1
Original file line numberDiff line numberDiff line change
@@ -8,4 +8,7 @@ REACT_APP_STAC_API=
88

99
## Theming
1010
# REACT_APP_THEME_PRIMARY_COLOR='#6A5ACD'
11-
# REACT_APP_THEME_SECONDARY_COLOR='#048A81'
11+
# REACT_APP_THEME_SECONDARY_COLOR='#048A81'
12+
13+
## Don't set the public url here. Check the README.md file for more information
14+
# PUBLIC_URL= Do not set here

packages/client/README.md

+10
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ Some client options are controlled by environment variables. These are:
1414
## Title and description of the app for metadata
1515
APP_TITLE
1616
APP_DESCRIPTION
17+
PUBLIC_URL
1718
1819
# API
1920
## If the app is being served in from a subfolder, the domain url must be set.
@@ -26,6 +27,15 @@ REACT_APP_THEME_PRIMARY_COLOR
2627
REACT_APP_THEME_SECONDARY_COLOR
2728
```
2829

30+
**Public URL**
31+
It is recommended to always set the `PUBLIC_URL` environment variable on a production build.
32+
If the app is being served from a subfolder, the `PUBLIC_URL` should include the subfolder path. **Do not include a trailing slash.**
33+
34+
For example, if the app is being served from `https://example.com/stac-manager`, the `PUBLIC_URL` should be set to `https://example.com/stac-manager`.
35+
36+
> [!IMPORTANT]
37+
> The `PUBLIC_URL` environment variable must be set before running the build script, and therefore the `.env` file cannot be used to set this variable.
38+
2939
You must provide a value for the `REACT_APP_STAC_API` environment variable. This should be the URL of the STAC API you wish to interact with.
3040

3141
If the `REACT_APP_STAC_BROWSER` environment variable is not set, [Radiant Earth's STAC Browser](https://radiantearth.github.io/stac-browser/) will be used by default, which will connect to the STAC API specified in `REACT_APP_STAC_API`.

packages/client/posthtml.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ module.exports = {
2828
locals: {
2929
appTitle: process.env.APP_TITLE,
3030
appDescription: process.env.APP_DESCRIPTION,
31-
baseurl: process.env.PUBLIC_URL || ''
31+
baseurl: process.env.PUBLIC_URL || '/'
3232
}
3333
}
3434
}

packages/client/src/App.tsx

+12-2
Original file line numberDiff line numberDiff line change
@@ -25,11 +25,21 @@ import CollectionDetail from './pages/CollectionDetail';
2525
import Sandbox from './pages/Sandbox';
2626
import { config } from './plugin-system/config';
2727

28+
let basename: string | undefined;
29+
if (process.env.PUBLIC_URL) {
30+
try {
31+
basename = new URL(process.env.PUBLIC_URL).pathname;
32+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
33+
} catch (error) {
34+
// no-op
35+
}
36+
}
37+
2838
export const App = () => (
2939
<ChakraProvider theme={theme}>
3040
<StacApiProvider apiUrl={process.env.REACT_APP_STAC_API!}>
3141
<PluginConfigProvider config={config}>
32-
<Router>
42+
<Router basename={basename}>
3343
<Container
3444
maxW='container.xl'
3545
minH='100vh'
@@ -46,7 +56,7 @@ export const App = () => (
4656
>
4757
<Flex gap={4} alignItems='center'>
4858
<Image
49-
src='/meta/icon-512.png'
59+
src={`${process.env.PUBLIC_URL}/meta/icon-512.png`}
5060
width={8}
5161
aspectRatio={1}
5262
borderRadius='md'

packages/client/src/pages/Home.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
import React from 'react';
2+
import { Navigate } from 'react-router-dom';
23

34
import { usePageTitle } from '../hooks';
4-
import { Navigate } from 'react-router-dom';
55

66
function Home() {
7-
usePageTitle('STAC Admin');
7+
usePageTitle(process.env.APP_TITLE!);
88

99
return <Navigate to='/collections' replace />;
1010
}

packages/client/tasks/build.mjs

+12-3
Original file line numberDiff line numberDiff line change
@@ -34,15 +34,23 @@ async function copyFiles() {
3434
log.info('📦 Copied static files to dist.');
3535
}
3636

37-
async function parcelServe() {
37+
async function parcelBuild() {
38+
const publicUrl = process.env.PUBLIC_URL || '/';
39+
40+
if (publicUrl && publicUrl !== '/') {
41+
log.warn(`🌍 Building using public URL: ${publicUrl}`);
42+
} else {
43+
log.warn(`🌍 Building without public URL`);
44+
}
45+
3846
const bundler = new Parcel({
3947
entries: `${__dirname}/../src/index.html`,
4048
defaultConfig: `${__dirname}/../.parcelrc`,
4149
cacheDir: `${__dirname}/../.parcel-cache`,
4250
mode: 'production',
4351
defaultTargetOptions: {
4452
distDir: `${__dirname}/../dist`,
45-
publicUrl: process.env.PUBLIC_URL || '/'
53+
publicUrl
4654
},
4755
additionalReporters: [
4856
{
@@ -58,8 +66,9 @@ async function parcelServe() {
5866
log.info(`✨ Built ${bundles.length} bundles in ${buildTime}ms!`);
5967
} catch (err) {
6068
log.warn(err.diagnostics);
69+
process.exit(1);
6170
}
6271
}
6372

6473
copyFiles();
65-
parcelServe();
74+
parcelBuild();
+121
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,121 @@
1+
/* global process */
2+
import path from 'path';
3+
import { fileURLToPath } from 'url';
4+
import fs from 'fs-extra';
5+
import log from 'fancy-log';
6+
7+
// Adapted into a script from: https://github.com/rafgraph/spa-github-pages/tree/gh-pages
8+
9+
const __filename = fileURLToPath(import.meta.url);
10+
const __dirname = path.dirname(__filename);
11+
12+
const baseUrl = process.env.PUBLIC_URL || '';
13+
14+
const pathIndex = path.join(__dirname, '../src/index.html');
15+
const path404 = path.join(__dirname, '../static/404.html');
16+
17+
async function main() {
18+
log.info('📦 Setting up single page apps on GitHub Pages.');
19+
20+
const has404 = await fs.pathExists(path404);
21+
22+
if (has404) {
23+
log.warn('📦 Found custom 404.html. Skipping setup.');
24+
process.exit(0);
25+
}
26+
27+
if (!baseUrl) {
28+
log.warn(
29+
'📦 Public URL not set. Assuming the app is deployed to the root.'
30+
);
31+
}
32+
33+
let segments = 0;
34+
if (baseUrl) {
35+
try {
36+
segments = new URL(baseUrl).pathname.split('/').length - 1;
37+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
38+
} catch (error) {
39+
// no-op
40+
}
41+
log.info(`📦 Using ${baseUrl} with ${segments} path segments.`);
42+
}
43+
44+
const templateScript = `<!-- Start Single Page Apps for GitHub Pages -->
45+
<script type="text/javascript">
46+
// Single Page Apps for GitHub Pages
47+
// MIT License
48+
// https://github.com/rafgraph/spa-github-pages
49+
// This script checks to see if a redirect is present in the query string,
50+
// converts it back into the correct url and adds it to the
51+
// browser's history using window.history.replaceState(...),
52+
// which won't cause the browser to attempt to load the new url.
53+
// When the single page app is loaded further down in this file,
54+
// the correct url will be waiting in the browser's history for
55+
// the single page app to route accordingly.
56+
(function(l) {
57+
if (l.search[1] === '/' ) {
58+
var decoded = l.search.slice(1).split('&').map(function(s) {
59+
return s.replace(/~and~/g, '&')
60+
}).join('?');
61+
window.history.replaceState(null, null,
62+
l.pathname.slice(0, -1) + decoded + l.hash
63+
);
64+
}
65+
}(window.location))
66+
</script>
67+
<!-- End Single Page Apps for GitHub Pages -->`;
68+
69+
// Write to index head.
70+
const index = await fs.readFile(pathIndex, 'utf8');
71+
const newIndex = index.replace('<head>', `<head>\n${templateScript}`);
72+
await fs.writeFile(pathIndex, newIndex);
73+
74+
const template404 = `<!DOCTYPE html>
75+
<html>
76+
<head>
77+
<meta charset="utf-8">
78+
<title>Single Page Apps for GitHub Pages</title>
79+
<script type="text/javascript">
80+
// Single Page Apps for GitHub Pages
81+
// MIT License
82+
// https://github.com/rafgraph/spa-github-pages
83+
// This script takes the current url and converts the path and query
84+
// string into just a query string, and then redirects the browser
85+
// to the new url with only a query string and hash fragment,
86+
// e.g. https://www.foo.tld/one/two?a=b&c=d#qwe, becomes
87+
// https://www.foo.tld/?/one/two&a=b~and~c=d#qwe
88+
// Note: this 404.html file must be at least 512 bytes for it to work
89+
// with Internet Explorer (it is currently > 512 bytes)
90+
91+
// If you're creating a Project Pages site and NOT using a custom domain,
92+
// then set pathSegmentsToKeep to 1 (enterprise users may need to set it to > 1).
93+
// This way the code will only replace the route part of the path, and not
94+
// the real directory in which the app resides, for example:
95+
// https://username.github.io/repo-name/one/two?a=b&c=d#qwe becomes
96+
// https://username.github.io/repo-name/?/one/two&a=b~and~c=d#qwe
97+
// Otherwise, leave pathSegmentsToKeep as 0.
98+
var pathSegmentsToKeep = ${segments};
99+
100+
var l = window.location;
101+
l.replace(
102+
l.protocol + '//' + l.hostname + (l.port ? ':' + l.port : '') +
103+
l.pathname.split('/').slice(0, 1 + pathSegmentsToKeep).join('/') + '/?/' +
104+
l.pathname.slice(1).split('/').slice(pathSegmentsToKeep).join('/').replace(/&/g, '~and~') +
105+
(l.search ? '&' + l.search.slice(1).replace(/&/g, '~and~') : '') +
106+
l.hash
107+
);
108+
109+
</script>
110+
</head>
111+
<body>
112+
</body>
113+
</html>`;
114+
115+
// Write to 404.html.
116+
await fs.writeFile(path404, template404);
117+
118+
log.info('✅ GitHub Pages setup complete.');
119+
}
120+
121+
main();

0 commit comments

Comments
 (0)