Files
Jellystat/src/pages/components/settings/webhooks.jsx
2025-05-26 10:11:24 +02:00

527 lines
21 KiB
JavaScript

import React, { useState, useEffect } from "react";
import axios from "../../../lib/axios_instance";
import { Form, Row, Col, Button, Spinner, Alert } from "react-bootstrap";
import InformationLineIcon from "remixicon-react/InformationLineIcon";
import { Tooltip } from "@mui/material";
import PropTypes from 'prop-types';
import Table from '@mui/material/Table';
import TableBody from '@mui/material/TableBody';
import TableCell from '@mui/material/TableCell';
import TableContainer from '@mui/material/TableContainer';
import TableHead from '@mui/material/TableHead';
import TableRow from '@mui/material/TableRow';
import { Trans } from "react-i18next";
import Loading from "../general/loading";
import ErrorBoundary from "../general/ErrorBoundary";
const token = localStorage.getItem('token');
// Modification du composant WebhookRow pour passer l'objet webhook complet
function WebhookRow(props) {
const { webhook, onEdit, onTest } = props;
return (
<React.Fragment>
<TableRow sx={{ '& > *': { borderBottom: 'unset' } }}>
<TableCell>{webhook.name}</TableCell>
<TableCell>{webhook.url}</TableCell>
<TableCell>{webhook.webhook_type || 'generic'}</TableCell>
<TableCell>{webhook.trigger_type}</TableCell>
<TableCell>
<span className={`badge ${webhook.enabled ? 'bg-success' : 'bg-secondary'}`}>
{webhook.enabled ? <Trans i18nKey={"ENABLED"} /> : <Trans i18nKey={"DISABLED"} />}
</span>
</TableCell>
<TableCell>
<div className="d-flex justify-content-end gap-2">
<Button size="sm" variant="outline-primary" onClick={() => onEdit(webhook)}>
<Trans i18nKey={"EDIT"} />
</Button>
<Button size="sm" variant="outline-secondary" onClick={() => onTest(webhook)}>
<Trans i18nKey={"SETTINGS_PAGE.TEST_NOW"} />
</Button>
</div>
</TableCell>
</TableRow>
</React.Fragment>
);
}
WebhookRow.propTypes = {
webhook: PropTypes.shape({
id: PropTypes.string.isRequired,
name: PropTypes.string.isRequired,
url: PropTypes.string.isRequired,
webhook_type: PropTypes.string,
trigger_type: PropTypes.string.isRequired,
enabled: PropTypes.bool.isRequired
}).isRequired,
onEdit: PropTypes.func.isRequired,
onTest: PropTypes.func.isRequired
};
function WebhooksSettings() {
const [webhooks, setWebhooks] = useState([]);
const [loading, setLoading] = useState(true);
const [saving, setSaving] = useState(false);
const [error, setError] = useState(null);
const [success, setSuccess] = useState(false);
const [currentWebhook, setCurrentWebhook] = useState({
name: 'Monthly Report of Most watched movies and series',
url: '',
enabled: false,
trigger_type: 'scheduled',
schedule: '0 9 1 * *',
method: 'POST',
webhook_type: 'discord'
});
const [eventWebhooks, setEventWebhooks] = useState({
playback_started: { exists: false, enabled: false },
playback_ended: { exists: false, enabled: false },
media_recently_added: { exists: false, enabled: false }
});
useEffect(() => {
const fetchWebhooks = async () => {
try {
const response = await axios.get('/webhooks', {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
});
if (response.data !== webhooks) {
setWebhooks(response.data);
await loadEventWebhooks();
}
if (loading) {
setLoading(false);
}
} catch (err) {
console.error("Error loading webhooks:", err);
if (loading) {
setLoading(false);
}
}
};
fetchWebhooks();
const intervalId = setInterval(fetchWebhooks, 1000 * 10);
return () => clearInterval(intervalId);
}, [webhooks.length]);
const handleInputChange = (e) => {
const { name, value } = e.target;
setCurrentWebhook(prev => ({ ...prev, [name]: value }));
};
const handleToggleEnabled = () => {
setCurrentWebhook(prev => ({ ...prev, enabled: !prev.enabled }));
};
const handleFormSubmit = async (e) => {
e.preventDefault();
try {
setSaving(true);
setError(null);
setSuccess(false);
if (!currentWebhook.url) {
setError("Webhook URL is required");
setSaving(false);
return;
}
if (currentWebhook.trigger_type === 'event' && !currentWebhook.event_type) {
setError("Event type is required for an event based webhook");
setSaving(false);
return;
}
let response;
if (currentWebhook.id) {
response = await axios.put(`/webhooks/${currentWebhook.id}`, currentWebhook, {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
}
});
} else {
response = await axios.post('/webhooks', currentWebhook, {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
}
});
}
const webhooksResponse = await axios.get('/webhooks', {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
});
setWebhooks(webhooksResponse.data);
await loadEventWebhooks();
setCurrentWebhook({
name: 'New Webhook',
url: '',
enabled: false,
trigger_type: 'scheduled',
schedule: '0 9 1 * *',
method: 'POST',
webhook_type: 'discord'
});
setSuccess("Webhook saved successfully!");
setSaving(false);
} catch (err) {
setError("Error while saving webhook " + (err.response?.data?.error || err.message));
setSaving(false);
}
};
const handleEdit = (webhook) => {
setCurrentWebhook(webhook);
};
const handleTest = async (webhook) => {
if (!webhook || !webhook.id) {
setError("Impossible to test the webhook: no webhook provided");
setLoading(false);
return;
}
try {
setLoading(true);
setError(null);
let endpoint = `/webhooks/${webhook.id}/test`;
if (webhook.trigger_type === 'scheduled' && webhook.schedule && webhook.schedule.includes('1 * *')) {
endpoint = `/webhooks/${webhook.id}/trigger-monthly`;
}
await axios.post(endpoint, {}, {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
}
});
setSuccess(`Webhook ${webhook.name} test triggered successfully!`);
setLoading(false);
} catch (err) {
setError("Error during the test of the webhook: " + (err.response?.data?.message || err.message));
setLoading(false);
}
};
const getEventWebhookStatus = (eventType) => {
return eventWebhooks[eventType]?.enabled || false;
};
const loadEventWebhooks = async () => {
try {
const eventTypes = ['playback_started', 'playback_ended', 'media_recently_added'];
const status = {};
eventTypes.forEach(eventType => {
const matchingWebhooks = webhooks.filter(
webhook => webhook.trigger_type === 'event' && webhook.event_type === eventType
);
status[eventType] = {
exists: matchingWebhooks.length > 0,
enabled: matchingWebhooks.some(webhook => webhook.enabled)
};
});
setEventWebhooks(status);
} catch (error) {
console.error('Error loading event webhook status:', error);
}
};
const toggleEventWebhook = async (eventType) => {
try {
setLoading(true);
setError(null);
const isCurrentlyEnabled = getEventWebhookStatus(eventType);
const matchingWebhooks = webhooks.filter(
webhook => webhook.trigger_type === 'event' && webhook.event_type === eventType
);
if (matchingWebhooks.length === 0 && !isCurrentlyEnabled) {
const newWebhook = {
name: `Notification - ${getEventDisplayName(eventType)}`,
url: '',
enabled: true,
trigger_type: 'event',
event_type: eventType,
method: 'POST',
webhook_type: 'discord'
};
setCurrentWebhook(newWebhook);
setLoading(false);
return;
}
for (const webhook of matchingWebhooks) {
await axios.put(`/webhooks/${webhook.id}`,
{ ...webhook, enabled: !isCurrentlyEnabled },
{
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
}
}
);
}
setEventWebhooks(prev => ({
...prev,
[eventType]: {
...prev[eventType],
enabled: !isCurrentlyEnabled
}
}));
const response = await axios.get('/webhooks', {
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
});
setWebhooks(response.data);
setLoading(false);
setSuccess(`Webhook for ${getEventDisplayName(eventType)} ${!isCurrentlyEnabled ? 'enabled' : 'disabled'} with success!`);
} catch (error) {
setError("Error while editing webhook: " + (error.response?.data?.error || error.message));
setLoading(false);
}
};
const getEventDisplayName = (eventType) => {
switch(eventType) {
case 'playback_started':
return 'Playback started';
case 'playback_ended':
return 'Playback ended';
case 'media_recently_added':
return 'New media added';
default:
return eventType;
}
};
if (loading && !webhooks.length) {
return <Loading />;
}
return (
<div className="webhooks-settings">
<h1 className="my-2">
<Trans i18nKey={"SETTINGS_PAGE.WEBHOOKS_CONFIGURATION"} />{" "}
<Tooltip title={<Trans i18nKey={"SETTINGS_PAGE.WEBHOOKS_TOOLTIP"} />}>
<span>
<InformationLineIcon />
</span>
</Tooltip>
</h1>
<ErrorBoundary>
{error && <Alert variant="danger" onClose={() => setError(null)} dismissible>{error}</Alert>}
{success && <Alert variant="success" onClose={() => setSuccess(false)} dismissible>
{typeof success === 'string' ? success : <Trans i18nKey={"SETTINGS_PAGE.WEBHOOK_SAVED"} />}
</Alert>}
<Form onSubmit={handleFormSubmit} className="settings-form">
<Form.Group as={Row} className="mb-3">
<Form.Label column sm={2}>
<Trans i18nKey={"SETTINGS_PAGE.WEBHOOK_NAME"} />
</Form.Label>
<Col sm={10}>
<Form.Control
type="text"
name="name"
value={currentWebhook.name}
onChange={handleInputChange}
required
/>
</Col>
</Form.Group>
<Form.Group as={Row} className="mb-3">
<Form.Label column sm={2}>
<Trans i18nKey={"SETTINGS_PAGE.DISCORD_WEBHOOK_URL"} />
</Form.Label>
<Col sm={10}>
<Form.Control
type="text"
name="url"
value={currentWebhook.url}
onChange={handleInputChange}
placeholder="https://discord.com/api/webhooks/..."
required
/>
</Col>
</Form.Group>
<Form.Group as={Row} className="mb-3">
<Form.Label column sm={2}>
<Trans i18nKey={"SETTINGS_PAGE.WEBHOOK_TYPE"} />
</Form.Label>
<Col sm={10}>
<Form.Select
name="webhook_type"
value={currentWebhook.webhook_type}
onChange={handleInputChange}
>
<option value="discord">Discord</option>
<option value="generic">Générique</option>
</Form.Select>
</Col>
</Form.Group>
<Form.Group as={Row} className="mb-3">
<Col sm={{ span: 10, offset: 2 }}>
<Form.Check
type="switch"
id="webhook-enabled"
label={<Trans i18nKey={"SETTINGS_PAGE.ENABLE_WEBHOOK"} />}
checked={currentWebhook.enabled}
onChange={handleToggleEnabled}
/>
</Col>
</Form.Group>
<Form.Group as={Row}>
<Col sm={{ span: 10, offset: 2 }}>
<Button
variant="primary"
type="submit"
disabled={saving}
>
{saving ? <Spinner size="sm" animation="border" /> : currentWebhook.id ? <Trans i18nKey={"UPDATE"} /> : <Trans i18nKey={"SAVE"} />}
</Button>
</Col>
</Form.Group>
</Form>
{/* Ajout de la section pour les webhooks événementiels */}
<div className="event-webhooks mt-4 mb-4">
<h3 className="my-3">
<Trans i18nKey={"SETTINGS_PAGE.EVENT_WEBHOOKS"} />
<Tooltip title={<Trans i18nKey={"SETTINGS_PAGE.EVENT_WEBHOOKS_TOOLTIP"} />}>
<span className="ms-2">
<InformationLineIcon />
</span>
</Tooltip>
</h3>
<Row className="g-4">
<Col md={4}>
<div className="border rounded p-3 h-25">
<div className="d-flex justify-content-between align-items-center mb-2">
<h5><Trans i18nKey={"SETTINGS_PAGE.PLAYBACK_STARTED"} /></h5>
<Form.Check
type="switch"
id="playback-started-enabled"
checked={getEventWebhookStatus('playback_started')}
onChange={() => toggleEventWebhook('playback_started')}
/>
</div>
<p className="small">
Send a webhook notification when a user starts watching a media
</p>
</div>
</Col>
<Col md={4}>
<div className="border rounded p-3 h-25">
<div className="d-flex justify-content-between align-items-center mb-2">
<h5><Trans i18nKey={"SETTINGS_PAGE.PLAYBACK_ENDED"} /></h5>
<Form.Check
type="switch"
id="playback-ended-enabled"
checked={getEventWebhookStatus('playback_ended')}
onChange={() => toggleEventWebhook('playback_ended')}
/>
</div>
<p className="small">
Send a webhook notification when a user finishes watching a media
</p>
</div>
</Col>
<Col md={4}>
<div className="border rounded p-3 h-25">
<div className="d-flex justify-content-between align-items-center mb-2">
<h5><Trans i18nKey={"SETTINGS_PAGE.MEDIA_ADDED"} /></h5>
<Form.Check
type="switch"
id="media-recently-added-enabled"
checked={getEventWebhookStatus('media_recently_added')}
onChange={() => toggleEventWebhook('media_recently_added')}
/>
</div>
<p className="small">
Send a webhook notification when new media is added to the library
</p>
</div>
</Col>
</Row>
</div>
<TableContainer className='rounded-2 mt-4'>
<Table aria-label="webhooks table">
<TableHead>
<TableRow>
<TableCell><Trans i18nKey={"SETTINGS_PAGE.NAME"} /></TableCell>
<TableCell><Trans i18nKey={"SETTINGS_PAGE.URL"} /></TableCell>
<TableCell><Trans i18nKey={"SETTINGS_PAGE.TYPE"} /></TableCell>
<TableCell><Trans i18nKey={"SETTINGS_PAGE.TRIGGER"} /></TableCell>
<TableCell><Trans i18nKey={"SETTINGS_PAGE.STATUS"} /></TableCell>
<TableCell></TableCell>
</TableRow>
</TableHead>
<TableBody>
{webhooks.map((webhook) => (
<WebhookRow
key={webhook.id}
webhook={webhook}
onEdit={handleEdit}
onTest={handleTest}
/>
))}
{webhooks.length === 0 && (
<TableRow>
<TableCell colSpan={6} style={{ textAlign: "center", fontStyle: "italic", color: "grey", height: "200px" }}>
<Trans i18nKey={"ERROR_MESSAGES.NO_WEBHOOKS"} />
</TableCell>
</TableRow>
)}
</TableBody>
</Table>
</TableContainer>
</ErrorBoundary>
</div>
);
}
export default WebhooksSettings;