mobile styles and info panel

This commit is contained in:
ansible user/allowed to read system logs 2023-08-07 13:52:15 -07:00
parent 48028193cc
commit bcc032413c
20 changed files with 280 additions and 56 deletions

View file

@ -1,26 +1,68 @@
import React from 'react'; import React, {useState} from 'react';
import { createTheme, ThemeProvider } from '@mui/material/styles'; import { createTheme, ThemeProvider } from '@mui/material/styles';
import { yellow } from "@mui/material/colors"; import { yellow } from "@mui/material/colors";
import GitHubIcon from '@mui/icons-material/GitHub';
import ModePanel from "./components/modePanel";
import Button from "@mui/material/Button";
import IconButton from "@mui/material/IconButton";
import InfoModal from "./components/modals/infoModal";
import {Info} from "@mui/icons-material";
const REPO_URL = "https://github.com/sneakers-the-rat/chatbridge"
const theme = createTheme({ const theme = createTheme({
palette:{ palette:{
primary: yellow, primary: yellow,
mode: "dark" mode: "dark",
} }
}) })
import ModePanel from "./components/modePanel";
function App() { function App() {
const [infoOpen, setInfoOpen] = useState(false);
const handleInfo = () => {
setInfoOpen(!infoOpen)
}
return ( return (
<ThemeProvider theme={theme}> <ThemeProvider theme={theme}>
<div className={"App-Container"}> <div className={"App-Container"}>
<div className="App"> <div className="App">
<div className={"App-header-bar"}>
<header className="App-header"> <header className="App-header">
ChatBridge ChatBridge
</header> </header>
<div className={"header-padding"}/>
<Button
className={"App-header-item"}
variant="contained"
aria-label={"source code"}
color={"primary"}
onClick={handleInfo}
>
INFO
</Button>
<Button
variant="contained"
aria-label={"source code"}
color={"primary"}
href={REPO_URL}
>
<GitHubIcon/>
</Button>
</div>
<ModePanel/> <ModePanel/>
<InfoModal open={infoOpen} setOpen={setInfoOpen}/>
</div> </div>
</div> </div>
</ThemeProvider> </ThemeProvider>

View file

@ -3,9 +3,7 @@ export const getBridgeByStateToken = (callback: CallableFunction) => {
fetch('api/bridge') fetch('api/bridge')
.then(res => res.json()) .then(res => res.json())
.then((res) => { .then((res) => {
console.log('bridge result', res)
if (res.status === 'success'){ if (res.status === 'success'){
console.log('successful get bridge')
callback(res.data) callback(res.data)
} }
}) })

View file

@ -22,8 +22,8 @@ export default function GroupPanel({
<TableHead> <TableHead>
<TableRow> <TableRow>
<TableCell>Group</TableCell> <TableCell>Group</TableCell>
<TableCell align="right">Created At</TableCell>
<TableCell align="right">Invite Token</TableCell> <TableCell align="right">Invite Token</TableCell>
<TableCell align="right">Created At</TableCell>
<TableCell align="right">Delete</TableCell> <TableCell align="right">Delete</TableCell>
</TableRow> </TableRow>
</TableHead> </TableHead>

View file

@ -32,8 +32,8 @@ export default function GroupRow(
<TableCell component="th" scope="row"> <TableCell component="th" scope="row">
{name} {name}
</TableCell> </TableCell>
<TableCell align="right">{created_at}</TableCell>
<TableCell align="right">{invite_token}</TableCell> <TableCell align="right">{invite_token}</TableCell>
<TableCell align="right">{created_at}</TableCell>
<TableCell align="right"> <TableCell align="right">
<IconButton <IconButton
onClick={handleDeleteGroup} onClick={handleDeleteGroup}

View file

@ -36,7 +36,7 @@ const JoinChannel = ({
useEffect(() => { useEffect(() => {
if (bridge){ if (bridge && !channels){
switch(platform){ switch(platform){
case "Slack": case "Slack":
getSlackChannels(setChannels) getSlackChannels(setChannels)

View file

@ -59,11 +59,11 @@ export const JoinForm = ({group}: JoinFormProps) => {
return ( return (
<> <>
<header className={'section-header'}> <header className={'section-header'}>
Joining group: <code>{group.name}</code> <code>{group.name}</code>
</header> </header>
<JoinStep <JoinStep
title={"1) Login"} title={"1) Login"}
details={"Select your chat platform"} // details={"Select your chat platform"}
id={'login'} id={'login'}
completed={stepComplete.login} completed={stepComplete.login}
> >
@ -78,7 +78,7 @@ export const JoinForm = ({group}: JoinFormProps) => {
</JoinStep> </JoinStep>
<JoinStep <JoinStep
title={"2) Configure Bridge"} title={"2) Configure Bridge"}
details={"Settings for all channels bridged from this platform"} // details={"Settings for all channels bridged from this platform"}
id={'bridge'} id={'bridge'}
disabled={!stepComplete.login} disabled={!stepComplete.login}
completed={stepComplete.bridge} completed={stepComplete.bridge}
@ -92,7 +92,7 @@ export const JoinForm = ({group}: JoinFormProps) => {
</JoinStep> </JoinStep>
<JoinStep <JoinStep
title={"3) Select a channel!"} title={"3) Select a channel!"}
details={"The bot will join :)"} // details={"The bot will join :)"}
id={'channel'} id={'channel'}
disabled={!stepComplete.login} disabled={!stepComplete.login}
completed={stepComplete.channel} completed={stepComplete.channel}

View file

@ -29,7 +29,6 @@ export const JoinGroup = ({
setAuthError(false); setAuthError(false);
setErrorText(''); setErrorText('');
setGroup(response.data) setGroup(response.data)
console.log(response)
} }
} }

View file

@ -30,18 +30,31 @@ export function JoinStep(
return( return(
<Accordion <Accordion
disabled={disabled} disabled={disabled}
id={id}
> >
<AccordionSummary <AccordionSummary
expandIcon={<ExpandMoreIcon />} expandIcon={<ExpandMoreIcon />}
aria-controls="panel1bh-content" aria-controls="panel1bh-content"
id="panel1bh-header" id="panel1bh-header"
> >
<Typography sx={{ width: '33%', flexShrink: 0 }}> <Typography sx={details === '' ?
{
width:'66%',
flexGrow: 1
}
:
{
width:'33%',
flexShrink: 0
}
}>
{ title } { title }
</Typography> </Typography>
{details !== '' ?
<Typography sx={{ color: 'text.secondary', flexGrow: 1 }}> <Typography sx={{ color: 'text.secondary', flexGrow: 1 }}>
{ details } { details }
</Typography> </Typography>
: undefined}
{ {
completed ? completed ?
<TaskAltIcon color={"success"}/> <TaskAltIcon color={"success"}/>
@ -50,9 +63,7 @@ export function JoinStep(
} }
</AccordionSummary> </AccordionSummary>
<AccordionDetails> <AccordionDetails>
<Typography>
{ children } { children }
</Typography>
</AccordionDetails> </AccordionDetails>
</Accordion> </Accordion>
) )

View file

@ -0,0 +1,82 @@
import * as React from "react";
import Dialog from '@mui/material/Dialog';
import DialogTitle from '@mui/material/DialogTitle';
import DialogContent from '@mui/material/DialogContent';
import DialogContentText from "@mui/material/DialogContentText";
export interface InfoModalProps {
open: boolean
setOpen: CallableFunction
}
const InfoModal = ({
open,
setOpen
}: InfoModalProps) => {
const handleClose = () => {
setOpen(false)
}
return(
<Dialog
open={open}
onClose={handleClose}
scroll={'body'}
>
<DialogTitle>
ChatBridge INFO
</DialogTitle>
<DialogContent>
<DialogContentText>
ChatBridge is a tool for making groupchats that span multiple chat protocols and platforms.
It is a web frontend for <a href={"https://github.com/42wim/matterbridge"}>matterbridge</a> and manages the API logins for platforms like Slack and Discord to lower the barriers to
bridging.
<br/><br/>
This is ALPHA software and functionality is not guaranteed! The eventual goal is to make <a href={"https://a.gup.pe/"}>guppe groups</a> for proprietary chat platforms, where anyone can create and join groups with just an invite token.
For now, creating groups is limited to an administrator token until we can be more sure about the security and performance demands of running many matterbridge processes.
</DialogContentText>
<h3>Security & Privacy Notes</h3>
<DialogContentText>
This tool is made by privacy advocates and activists. The developers have no interest in storing or monetizing your information, full stop. We will <span style={{"fontStyle":"italic"}}>never</span> abuse the app tokens created by logging into the ChatBridge App for anything except configuring the underlying matterbridge processes
<br/><br/>
To be as transparent as possible about the operation of the service:
</DialogContentText>
<h4>What is stored</h4>
<DialogContentText>
<ul>
<li>Slack/Discord: App login tokens that are generated for your workspace/server</li>
<li>Limited metadata about your bridged chat, specifically its name and unique ID (usually these are considered public anyway), and the short label you provide to the service</li>
<li>The names of channels you have bridged</li>
</ul>
</DialogContentText>
<h4>What is NOT stored</h4>
<DialogContentText>
<ul>
<li>Message content and metadata of any kind, even in debugging logs</li>
<li>No other metadata about your bridged chat except for that listed above - channel lists in the interface are requested and discarded during the API call</li>
<li>The bot access tokens can NOT access DMs or private channels</li>
</ul>
</DialogContentText>
<h4>Security of Data at Rest</h4>
<DialogContentText>
Matterbridge relies on <code>.toml</code> files that contain the app tokens in plain text. This sets a hard limit on how secure the data can be at rest - ie. encryption at rest is not possible. The tokens that are granted to ChatBridge CAN be used to exfiltrate chat history of public channels for slack and discord if they were to be lost in a data breach. We therfore do NOT recommend you use ChatBridge in a context where the contents of your public channels or the membership in your chatroom becoming public could pose a risk to your members.
<br/><br/>
For the main development instance, the service is run as its own user, and the configuration files are stored such that only that user can read them. For the data to be breached, an attacker would need to compromise a root SSH key.
<br/><br/>
For information accessed through the chatbridge interface, we authenticate using ephemeral signed cookies that expire 24 hours after they are issued.
<br/><br/>
We of course can make no guarantee about security of data at any other instances that deploy ChatBridge.
<br/><br/>
To revoke access to ChatBridge at any time, you can uninstall the app from your slack workspace or discord server - this makes the access keys obsolete and will require them to be reissued if you choose to rejoin ChatBridge.
</DialogContentText>
</DialogContent>
</Dialog>
)
}
export default InfoModal

View file

@ -30,8 +30,8 @@ export default function ModePanel() {
onChange={handleChange} onChange={handleChange}
selectionFollowsFocus> selectionFollowsFocus>
{/*<TabsList className={"TabsList"}>*/} {/*<TabsList className={"TabsList"}>*/}
<Tab label={"Join Group"}></Tab> <Tab className={"Tab"} label={"Join Group"}></Tab>
<Tab label={"Create Group"}></Tab> <Tab className={"Tab"} label={"Create Group"}></Tab>
{/*</TabsList>*/} {/*</TabsList>*/}
</StyledTabs> </StyledTabs>

View file

@ -14,7 +14,8 @@ export default function TabPanel(props) {
{...other} {...other}
> >
{value === index && ( {value === index && (
<Box sx={{ p: 3 }}> <Box
className={"TabPanel"}>
{children} {children}
</Box> </Box>
)} )}

View file

@ -11,6 +11,8 @@
//} //}
border: 2px solid $color-primary; border: 2px solid $color-primary;
height:100%; height:100%;
box-sizing: border-box;
overflow-y: auto;
} }
.App-Container { .App-Container {
@ -23,6 +25,11 @@
width:100%; width:100%;
height:100%; height:100%;
box-sizing: border-box; box-sizing: border-box;
@media (max-width: $breakpoint-mobile) {
padding: 0;
box-sizing: border-box;
}
} }
@ -37,17 +44,49 @@
} }
} }
.App-header { .App-header-bar{
text-align: center;
margin: {
top: 1em;
}
display: flex; display: flex;
flex-direction: column; flex-direction: row;
align-items: center; padding: 0.5em 0.5em;
justify-content: center; gap: 0.5em;
font-size: calc(10px + 2vmin); font: {
color: white; family: "Courier", monospace;
}
background: repeating-linear-gradient(
45deg,
$color-primary,
$color-primary 20px,
$color-background 20px,
$color-background 40px
);
.header-padding {
flex-grow: 1;
}
.App-header-item {
font: {
family: "Courier", monospace;
}
}
}
.App-header {
font: {
size: calc(10px + 2vmin);
weight: bold;
}
margin: auto;
color: $color-background;
background-color: $color-primary;
padding: 0.25em;
text-align: left;
//border-radius: 4px;
//box-shadow: 0px 2px 4px -1px rgba(0,0,0,0.2),
// 0px 4px 5px 0px rgba(0,0,0,0.14),
// 0px 1px 10px 0px rgba(0,0,0,0.12);
} }
.App-link { .App-link {

View file

@ -18,3 +18,9 @@ code {
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
monospace; monospace;
} }
//@media (max-width: $breakpoint-mobile){
// .MuiBox-root {
// padding: 6px !important;
// }
//}

View file

@ -4,6 +4,10 @@
align-items: flex-start; align-items: flex-start;
gap: 2em; gap: 2em;
@media (max-width: $breakpoint-mobile) {
gap: 0.5em;
}
&>div { &>div {
flex-basis: 50%; flex-basis: 50%;
} }
@ -39,7 +43,9 @@
gap: 2em; gap: 2em;
align-items: flex-start; align-items: flex-start;
@media (max-width: $breakpoint-mobile){
gap: 0.5em;
}
.Input { .Input {
flex-basis: 70%; flex-basis: 70%;

View file

@ -1,7 +1,10 @@
.ModePanel { .ModePanel {
max-width: 80%; //max-width: 80%;
margin: auto; margin: auto;
font: {
family: "Courier", monospace;
}
.TabsList { .TabsList {
min-width: 400px; min-width: 400px;
@ -19,12 +22,24 @@
justify-content: center; justify-content: center;
align-content: space-between; align-content: space-between;
.Tab.Mui-selected {
background-color: $color-primary;
color: $color-background;
}
}
.Tab { .Tab {
font-family: 'IBM Plex Sans', sans-serif; //font-size: 1000px;
font: {
family: 'Courier', monospace;
size: 1.2em;
}
color: white; color: white;
cursor: pointer; cursor: pointer;
font-size: 1.5rem; //font-size: 1.5rem;
font-weight: bold; //font-weight: bold;
background-color: transparent; background-color: transparent;
width: 100%; width: 100%;
line-height: 1.5; line-height: 1.5;
@ -35,11 +50,12 @@
justify-content: center; justify-content: center;
border: none; border: none;
} }
.Tab.Mui-selected {
background-color: $color-primary;
color: $color-background;
} }
.TabPanel {
padding: 24px;
@media (max-width: $breakpoint-mobile){
padding: 12px 12px;
} }
} }

View file

@ -10,6 +10,16 @@
} }
} }
p a {
background-color: $color-primary;
color: $color-background;
padding: 0.2em;
font: {
family: "Courier", monospace;
}
border-radius: 3px;
}
code { code {
font: { font: {
family: 'Courier', 'Courier New', monospace; family: 'Courier', 'Courier New', monospace;

View file

@ -1,7 +1,9 @@
$color-primary: #DED03A; $color-primary: #ffeb3b;
$color-primary-darker: scale-color($color-primary, $lightness: -10%); $color-primary-darker: scale-color($color-primary, $lightness: -10%);
$color-primary-background: adjust-color($color-primary, $alpha: -0.9, $saturation: -50%); $color-primary-background: adjust-color($color-primary, $alpha: -0.9, $saturation: -50%);
$color-primary-border: adjust-color($color-primary, $alpha: -0.9, $lightness: 50%, $saturation: -50%); $color-primary-border: adjust-color($color-primary, $alpha: -0.9, $lightness: 50%, $saturation: -50%);
$color-secondary: #A167A5; $color-secondary: #A167A5;
$color-background: #1C1B22; $color-background: #1C1B22;
$color-text: white; $color-text: white;
$breakpoint-mobile: 600px;

10
client/src/types.d.ts vendored
View file

@ -3,3 +3,13 @@ declare module "*.svg" {
const content: any; const content: any;
export default content; export default content;
} }
// declare module '@mui/material/styles' {
// interface Palette {
// paleYellow: Palette['primary'];
// }
//
// interface PaletteOptions {
// paleYellow?: PaletteOptions['primary'];
// }
// }

View file

@ -15,7 +15,7 @@ const scopes = ['bot', 'channels:write', 'channels:write.invites', 'chat:write:b
const slackBridgeRepository = AppDataSource.getRepository(SlackBridge) const slackBridgeRepository = AppDataSource.getRepository(SlackBridge)
const groupRepository = AppDataSource.getRepository(Group) const groupRepository = AppDataSource.getRepository(Group)
const SLACK_COOKIE_NAME = "slack-oauth-state"; // const SLACK_COOKIE_NAME = "slack-oauth-state";
const slackConfig = config.get<slackConfigType>('slackConfig'); const slackConfig = config.get<slackConfigType>('slackConfig');
@ -45,7 +45,7 @@ export const SlackInstallLinkHandler = async(
state_token state_token
); );
res.cookie(SLACK_COOKIE_NAME, state_token, { maxAge: 60*5 }) // res.cookie(SLACK_COOKIE_NAME, state_token, { maxAge: 60*5 })
res.status(200).json({ res.status(200).json({
status: 'success', status: 'success',
data: { data: {
@ -78,7 +78,9 @@ export const SlackCallbackHandler = async(
// check if we have an entity // check if we have an entity
let bridge = await slackBridgeRepository.findOneBy({Token: installation.bot.token}) let bridge = await slackBridgeRepository.findOneBy({Token: installation.bot.token})
logger.debug('found existing bridge %s', bridge)
if (!bridge){ if (!bridge){
logger.debug('creating new bridge')
bridge = await slackBridgeRepository.create(bridge_data); bridge = await slackBridgeRepository.create(bridge_data);
await slackBridgeRepository.save(bridge); await slackBridgeRepository.save(bridge);
logger.debug('created bridge') logger.debug('created bridge')

View file

@ -19,7 +19,7 @@ export const cookieMiddleware = cookieSession({
name: 'session', name: 'session',
keys: [cookieConfig.key1, cookieConfig.key2], keys: [cookieConfig.key1, cookieConfig.key2],
signed: true, signed: true,
maxAge: 24*60*60*1000 // 24 hours
}) })
export const requireAdmin = (req: Request, res: Response, next: NextFunction) => { export const requireAdmin = (req: Request, res: Response, next: NextFunction) => {