Skip to content

Commit 76a7790

Browse files
committed
init: timezone plugin example
Signed-off-by: Mayur Deshmukh <[email protected]>
0 parents  commit 76a7790

20 files changed

+19543
-0
lines changed

.editorconfig

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
root = true
2+
3+
[*]
4+
end_of_line = lf
5+
insert_final_newline = true
6+
7+
[*.{js,json,yml}]
8+
charset = utf-8
9+
indent_style = space
10+
indent_size = 2

.gitattributes

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
/.yarn/** linguist-vendored
2+
/.yarn/releases/* binary
3+
/.yarn/plugins/**/* binary
4+
/.pnp.* binary linguist-generated

.gitignore

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
.yarn/*
2+
!.yarn/patches
3+
!.yarn/plugins
4+
!.yarn/releases
5+
!.yarn/sdks
6+
!.yarn/versions
7+
8+
node_modules/
9+
dist/
10+
dist-scalprum/
11+
dist-types/
12+
dynamic-plugins-root/*
13+
!dynamic-plugins-root/.gitkeep
14+
15+
.env

.yarnrc.yml

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
nodeLinker: node-modules

README.md

+52
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
# devconf-2024-workshop
2+
3+
The workshop contents for the DevConf.cz 2024.
4+
5+
## How to use
6+
7+
Please switch to the `start-here` branch for the starting point.
8+
9+
You can refer the `main` branch for the final stage.
10+
11+
## Getting Started
12+
13+
You can refer the dynamic-plugins readme for reference:\
14+
https://github.com/janus-idp/backstage-showcase/blob/main/showcase-docs/dynamic-plugins.md
15+
16+
## Running on local environments
17+
18+
1. Install dependencies
19+
20+
```
21+
yarn install
22+
```
23+
24+
2. Generate types
25+
26+
```
27+
yarn tsc
28+
```
29+
30+
3. Build the plugin
31+
32+
```
33+
yarn workspace demo-timezone-plugin build
34+
```
35+
36+
4. Export the plugin as a dynamic plugin
37+
38+
```
39+
yarn workspace demo-timezone-plugin export-dynamic
40+
```
41+
42+
Now that the plugin has been exported, you can start the backstage instance with podman
43+
44+
```
45+
podman compose up -d rhdh
46+
```
47+
48+
## Contributors
49+
50+
- Mayur Deshmukh ([@deshmukhmayur](https://github.com/deshmukhmayur))
51+
- Rigin Oommen ([@riginoommen](https://github.com/riginoommen))
52+
- Yash Oswal ([@yashoswalyo](https://github.com/yashoswalyo))

app-config.yaml

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
app:
2+
title: Backstage Demo
3+
baseUrl: "http://localhost:7007/"
4+
5+
branding:
6+
fullLogo: ""
7+
iconLogo: ""
8+
theme:
9+
light:
10+
primaryColor: ${PRIMARY_LIGHT_COLOR}
11+
headerColor1: '#be0000'
12+
headerColor2: '#f56d6d'
13+
navigationIndicatorColor: '#be0000'
14+
dark:
15+
primaryColor: ${PRIMARY_DARK_COLOR}
16+
headerColor1: '#be0000'
17+
headerColor2: '#f56d6d'
18+
navigationIndicatorColor: '#be0000'
19+
20+
backend:
21+
baseUrl: http://localhost:7007/
22+
listen:
23+
port: 7007
24+
25+
auth:
26+
keys:
27+
- secret: 'demobackstagesecret'
28+
29+
catalog:
30+
locations:
31+
# Local example data, file locations are relative to the backend process, typically `packages/backend`
32+
- type: file
33+
target: ./catalog-info.example.yaml

catalog-info.example.yaml

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
apiVersion: backstage.io/v1alpha1
2+
kind: Component
3+
metadata:
4+
name: rhdh
5+
title: Red Hat Developer Hub
6+
description: |
7+
An enterprise-grade, open developer platform for building developer portals, containing a supported and opinionated framework.
8+
links:
9+
- title: Website
10+
url: http://developer.redhat.com/rhdh
11+
- title: Documentation
12+
url: https://backstage.io/docs
13+
- title: Learning Path
14+
url: https://developers.redhat.com/learn/openshift/install-and-configure-red-hat-developer-hub-and-explore-templating-basics
15+
annotations:
16+
demo/timezone: Asia/Kolkata
17+
spec:
18+
type: website
19+
owner: user:default/guest
20+
lifecycle: production

docker-compose.yaml

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
services:
2+
rhdh:
3+
image: quay.io/janus-idp/backstage-showcase:latest
4+
volumes:
5+
- ./dynamic-plugins-root:/opt/app-root/src/dynamic-plugins-root:z
6+
- ./app-config.yaml:/opt/app-root/src/app-config.example.production.yaml:z
7+
- ./catalog-info.example.yaml:/opt/app-root/src/catalog-info.example.yaml:z
8+
ports:
9+
- 7007:7007
10+
environment:
11+
- POSTGRES_HOST=postgres
12+
- POSTGRES_PORT=5432

dynamic-plugins-root/.gitkeep

Whitespace-only changes.

dynamic-plugins.default.yaml

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
dynamicPlugins:
2+
frontend:

package.json

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
{
2+
"name": "devconf-2024-workshop",
3+
"private": true,
4+
"engines": {
5+
"node": ">=20.12.2"
6+
},
7+
"scripts": {
8+
"tsc": "tsc"
9+
},
10+
"workspaces": {
11+
"packages": [
12+
"plugins/*"
13+
]
14+
},
15+
"dependencies": {
16+
"@backstage/cli": "^0.26.6"
17+
},
18+
"devDependencies": {
19+
"@testing-library/react": "^16.0.0",
20+
"react": "^18.3.1",
21+
"react-dom": "^18.3.1",
22+
"react-router-dom": "^6.23.1",
23+
"typescript": "^5.4.5"
24+
},
25+
"packageManager": "[email protected]"
26+
}

plugins/demo-timezone/dev/index.tsx

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import React from "react";
2+
import { createDevApp } from "@backstage/dev-utils";
3+
import { demoTimezonePlugin, Clock } from "../src/plugin";
4+
import { EntityProvider } from "@backstage/plugin-catalog-react";
5+
import Grid from "@mui/material/Grid";
6+
7+
const mockEntity = {
8+
apiVersion: 'backstage.io/v1alpha1',
9+
kind: 'Component',
10+
metadata: {
11+
name: 'Example Application',
12+
annotations: {
13+
'demo/timezone': 'Asia/Kolkata'
14+
}
15+
},
16+
spec: {
17+
type: 'service',
18+
owner: 'user:default/guest'
19+
}
20+
}
21+
22+
createDevApp()
23+
.registerPlugin(demoTimezonePlugin)
24+
.addPage({
25+
title: "Demo Timezone Test",
26+
path: "/demo",
27+
element: (
28+
<EntityProvider entity={mockEntity}>
29+
<Grid item md={6}>
30+
<Clock />
31+
</Grid>
32+
</EntityProvider>
33+
),
34+
})
35+
.render();

plugins/demo-timezone/package.json

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
{
2+
"name": "demo-timezone-plugin",
3+
"version": "1.0.0",
4+
"description": "",
5+
"main": "index.js",
6+
"backstage": {
7+
"role": "frontend-plugin"
8+
},
9+
"scripts": {
10+
"start": "backstage-cli package start",
11+
"build": "backstage-cli package build"
12+
},
13+
"keywords": [],
14+
"author": "",
15+
"license": "ISC",
16+
"dependencies": {
17+
"@backstage/catalog-model": "^1.5.0",
18+
"@backstage/cli": "^0.26.6",
19+
"@backstage/core-components": "^0.14.7",
20+
"@backstage/core-plugin-api": "^1.9.2",
21+
"@backstage/plugin-catalog-react": "^1.12.0",
22+
"@backstage/theme": "^0.5.5",
23+
"@mui/material": "^5.15.20",
24+
"dayjs": "^1.11.11"
25+
},
26+
"peerDependencies": {
27+
"react": "^16.13.1 || ^17.0.0 || ^18.0.0"
28+
},
29+
"devDependencies": {
30+
"@backstage/dev-utils": "^1.0.32"
31+
},
32+
"files": [
33+
"dist"
34+
]
35+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
import React from 'react';
2+
import { useEntity } from '@backstage/plugin-catalog-react';
3+
import dayjs from 'dayjs';
4+
import utc from 'dayjs/plugin/utc';
5+
import timezone from 'dayjs/plugin/timezone';
6+
import { getTimezone } from '../utils';
7+
import { Card, CardContent, Typography } from '@mui/material';
8+
dayjs.extend(utc);
9+
dayjs.extend(timezone);
10+
11+
export function Clock() {
12+
const { entity } = useEntity();
13+
const timezone = getTimezone(entity);
14+
15+
const time = dayjs();
16+
const formattedTime = time.tz(timezone ?? undefined).format('hh:mm:ss a Z');
17+
18+
return (
19+
<Card>
20+
<CardContent>
21+
<Typography>
22+
Time in {timezone}:
23+
</Typography>
24+
<Typography variant='h6'>
25+
{formattedTime}
26+
</Typography>
27+
</CardContent>
28+
</Card>
29+
);
30+
}

plugins/demo-timezone/src/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export { demoTimezonePlugin, Clock } from './plugin';

plugins/demo-timezone/src/plugin.ts

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import {
2+
createPlugin,
3+
createRoutableExtension,
4+
} from "@backstage/core-plugin-api";
5+
import { rootRouteRef } from "./routes";
6+
7+
export const demoTimezonePlugin = createPlugin({
8+
id: "demo-timezone",
9+
routes: {
10+
root: rootRouteRef,
11+
},
12+
});
13+
14+
export const Clock = demoTimezonePlugin.provide(
15+
createRoutableExtension({
16+
name: "Clock",
17+
component: () => import("./components/Clock").then((m) => m.Clock),
18+
mountPoint: rootRouteRef,
19+
})
20+
);

plugins/demo-timezone/src/routes.ts

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
import { createRouteRef } from "@backstage/core-plugin-api";
2+
3+
export const rootRouteRef = createRouteRef({
4+
id: 'demo-timezone',
5+
});

plugins/demo-timezone/src/utils.ts

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
import { Entity } from "@backstage/catalog-model";
2+
3+
export function getTimezone(entity: Entity) {
4+
return entity.metadata.annotations?.['demo/timezone'];
5+
}

tsconfig.json

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
{
2+
"extends": "@backstage/cli/config/tsconfig.json",
3+
"include": [
4+
"plugins/*/src"
5+
],
6+
"exclude": ["node_modules"],
7+
"compilerOptions": {
8+
"outDir": "dist-types",
9+
"rootDir": ".",
10+
"jsx": "react",
11+
"skipLibCheck": true
12+
}
13+
}

0 commit comments

Comments
 (0)