mirror of
https://github.com/ajnart/homarr.git
synced 2025-11-10 07:25:48 +01:00
✨ Add RSS widget
This commit is contained in:
@@ -49,6 +49,7 @@
|
|||||||
"dockerode": "^3.3.2",
|
"dockerode": "^3.3.2",
|
||||||
"fily-publish-gridstack": "^0.0.13",
|
"fily-publish-gridstack": "^0.0.13",
|
||||||
"framer-motion": "^6.5.1",
|
"framer-motion": "^6.5.1",
|
||||||
|
"html-entities": "^2.3.3",
|
||||||
"i18next": "^21.9.1",
|
"i18next": "^21.9.1",
|
||||||
"js-file-download": "^0.4.12",
|
"js-file-download": "^0.4.12",
|
||||||
"next": "^13.1.6",
|
"next": "^13.1.6",
|
||||||
@@ -58,6 +59,7 @@
|
|||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
"react-simple-code-editor": "^0.13.1",
|
"react-simple-code-editor": "^0.13.1",
|
||||||
|
"rss-parser": "^3.12.0",
|
||||||
"sabnzbd-api": "^1.5.0",
|
"sabnzbd-api": "^1.5.0",
|
||||||
"uuid": "^8.3.2",
|
"uuid": "^8.3.2",
|
||||||
"yarn": "^1.22.19",
|
"yarn": "^1.22.19",
|
||||||
|
|||||||
20
public/locales/en/modules/rss.json
Normal file
20
public/locales/en/modules/rss.json
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"descriptor": {
|
||||||
|
"name": "RSS Widget",
|
||||||
|
"description": "Grabs the items from a RSS feed and displays them. Commonly used for online news",
|
||||||
|
"settings": {
|
||||||
|
"title": "Settings for RSS widget",
|
||||||
|
"rssFeedUrl": {
|
||||||
|
"label": "RSS feed url"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"card": {
|
||||||
|
"errors": {
|
||||||
|
"general": {
|
||||||
|
"title": "Unable to retrieve RSS feed",
|
||||||
|
"text": "There was a problem reaching out the the RSS feed. Make sure that you've configured the feed correctly and use a valid RSS url, that matches the official standard specification. After updating the feed, you may need to save your dashboard and refresh the page."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
10
src/hooks/widgets/rss/useGetRssFeed.tsx
Normal file
10
src/hooks/widgets/rss/useGetRssFeed.tsx
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { useQuery } from '@tanstack/react-query';
|
||||||
|
|
||||||
|
export const useGetRssFeed = (feedUrl: string) =>
|
||||||
|
useQuery({
|
||||||
|
queryKey: ['rss-feed', feedUrl],
|
||||||
|
queryFn: async () => {
|
||||||
|
const response = await fetch('/api/modules/rss');
|
||||||
|
return response.json();
|
||||||
|
},
|
||||||
|
});
|
||||||
93
src/pages/api/modules/rss/index.ts
Normal file
93
src/pages/api/modules/rss/index.ts
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
import Consola from 'consola';
|
||||||
|
|
||||||
|
import { getCookie } from 'cookies-next';
|
||||||
|
|
||||||
|
import { decode } from 'html-entities';
|
||||||
|
|
||||||
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
|
import Parser from 'rss-parser';
|
||||||
|
|
||||||
|
import { getConfig } from '../../../../tools/config/getConfig';
|
||||||
|
import { Stopwatch } from '../../../../tools/shared/stopwatch';
|
||||||
|
import { IRssWidget } from '../../../../widgets/rss/RssWidgetTile';
|
||||||
|
|
||||||
|
type CustomItem = {
|
||||||
|
'media:content': string;
|
||||||
|
enclosure: {
|
||||||
|
url: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const parser: Parser<any, CustomItem> = new Parser({
|
||||||
|
customFields: {
|
||||||
|
item: ['media:content', 'enclosure'],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export const Get = async (request: NextApiRequest, response: NextApiResponse) => {
|
||||||
|
const configName = getCookie('config-name', { req: request });
|
||||||
|
const config = getConfig(configName?.toString() ?? 'default');
|
||||||
|
|
||||||
|
const rssWidget = config.widgets.find((x) => x.id === 'rss') as IRssWidget | undefined;
|
||||||
|
|
||||||
|
if (
|
||||||
|
!rssWidget ||
|
||||||
|
!rssWidget.properties.rssFeedUrl ||
|
||||||
|
rssWidget.properties.rssFeedUrl.length < 1
|
||||||
|
) {
|
||||||
|
response.status(400).json({ message: 'required widget does not exist' });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Consola.info('Requesting RSS feed...');
|
||||||
|
const stopWatch = new Stopwatch();
|
||||||
|
const feed = await parser.parseURL(rssWidget.properties.rssFeedUrl);
|
||||||
|
Consola.info(`Retrieved RSS feed after ${stopWatch.getEllapsedMilliseconds()} milliseconds`);
|
||||||
|
|
||||||
|
const orderedFeed = {
|
||||||
|
...feed,
|
||||||
|
items: feed.items
|
||||||
|
.map((item: { title: any; content: any }) => ({
|
||||||
|
...item,
|
||||||
|
title: item.title ? decode(item.title) : undefined,
|
||||||
|
content: decode(item.content),
|
||||||
|
enclosure: createEnclosure(item),
|
||||||
|
}))
|
||||||
|
.sort((a: { pubDate: number }, b: { pubDate: number }) => {
|
||||||
|
if (!a.pubDate || !b.pubDate) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return a.pubDate - b.pubDate;
|
||||||
|
})
|
||||||
|
.slice(0, 20),
|
||||||
|
};
|
||||||
|
|
||||||
|
response.status(200).json({
|
||||||
|
feed: orderedFeed,
|
||||||
|
success: orderedFeed?.items !== undefined,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const createEnclosure = (item: any) => {
|
||||||
|
if (item.enclosure) {
|
||||||
|
return item.enclosure;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (item['media:content']) {
|
||||||
|
return {
|
||||||
|
url: item['media:content'].$.url,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async (request: NextApiRequest, response: NextApiResponse) => {
|
||||||
|
if (request.method === 'GET') {
|
||||||
|
return Get(request, response);
|
||||||
|
}
|
||||||
|
|
||||||
|
return response.status(405);
|
||||||
|
};
|
||||||
@@ -92,3 +92,8 @@
|
|||||||
height: 0px;
|
height: 0px;
|
||||||
min-height: 0px !important;
|
min-height: 0px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.scroll-area-w100 .mantine-ScrollArea-viewport > div:nth-of-type(1) {
|
||||||
|
width: 100%;
|
||||||
|
display: inherit !important;
|
||||||
|
}
|
||||||
@@ -29,6 +29,7 @@ export const dashboardNamespaces = [
|
|||||||
'modules/torrents-status',
|
'modules/torrents-status',
|
||||||
'modules/weather',
|
'modules/weather',
|
||||||
'modules/ping',
|
'modules/ping',
|
||||||
|
'modules/rss',
|
||||||
'modules/docker',
|
'modules/docker',
|
||||||
'modules/dashdot',
|
'modules/dashdot',
|
||||||
'modules/overseerr',
|
'modules/overseerr',
|
||||||
|
|||||||
11
src/tools/shared/stopwatch.ts
Normal file
11
src/tools/shared/stopwatch.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
export class Stopwatch {
|
||||||
|
private startTime: Date;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.startTime = new Date();
|
||||||
|
}
|
||||||
|
|
||||||
|
getEllapsedMilliseconds() {
|
||||||
|
return new Date().getTime() - this.startTime.getTime();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
import date from './date/DateTile';
|
|
||||||
import calendar from './calendar/CalendarTile';
|
import calendar from './calendar/CalendarTile';
|
||||||
import dashdot from './dashDot/DashDotTile';
|
import dashdot from './dashDot/DashDotTile';
|
||||||
import usenet from './useNet/UseNetTile';
|
import date from './date/DateTile';
|
||||||
import weather from './weather/WeatherTile';
|
|
||||||
import torrent from './torrent/TorrentTile';
|
|
||||||
import torrentNetworkTraffic from './download-speed/TorrentNetworkTrafficTile';
|
import torrentNetworkTraffic from './download-speed/TorrentNetworkTrafficTile';
|
||||||
|
import rss from './rss/RssWidgetTile';
|
||||||
|
import torrent from './torrent/TorrentTile';
|
||||||
|
import usenet from './useNet/UseNetTile';
|
||||||
import videoStream from './video/VideoStreamTile';
|
import videoStream from './video/VideoStreamTile';
|
||||||
|
import weather from './weather/WeatherTile';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
calendar,
|
calendar,
|
||||||
@@ -15,5 +16,6 @@ export default {
|
|||||||
'torrents-status': torrent,
|
'torrents-status': torrent,
|
||||||
dlspeed: torrentNetworkTraffic,
|
dlspeed: torrentNetworkTraffic,
|
||||||
date,
|
date,
|
||||||
|
rss,
|
||||||
'video-stream': videoStream,
|
'video-stream': videoStream,
|
||||||
};
|
};
|
||||||
|
|||||||
236
src/widgets/rss/RssWidgetTile.tsx
Normal file
236
src/widgets/rss/RssWidgetTile.tsx
Normal file
@@ -0,0 +1,236 @@
|
|||||||
|
import {
|
||||||
|
ActionIcon,
|
||||||
|
Badge,
|
||||||
|
Card,
|
||||||
|
Center,
|
||||||
|
createStyles,
|
||||||
|
Flex,
|
||||||
|
Group,
|
||||||
|
Image,
|
||||||
|
Loader,
|
||||||
|
LoadingOverlay,
|
||||||
|
MediaQuery,
|
||||||
|
ScrollArea,
|
||||||
|
Stack,
|
||||||
|
Text,
|
||||||
|
Title,
|
||||||
|
UnstyledButton,
|
||||||
|
} from '@mantine/core';
|
||||||
|
import { useElementSize } from '@mantine/hooks';
|
||||||
|
import {
|
||||||
|
IconBulldozer,
|
||||||
|
IconCalendarTime,
|
||||||
|
IconClock,
|
||||||
|
IconCopyright,
|
||||||
|
IconRefresh,
|
||||||
|
IconRss,
|
||||||
|
IconSpeakerphone,
|
||||||
|
} from '@tabler/icons';
|
||||||
|
import { useTranslation } from 'next-i18next';
|
||||||
|
import Link from 'next/link';
|
||||||
|
import { useState } from 'react';
|
||||||
|
import { useGetRssFeed } from '../../hooks/widgets/rss/useGetRssFeed';
|
||||||
|
import { sleep } from '../../tools/client/time';
|
||||||
|
import { defineWidget } from '../helper';
|
||||||
|
import { IWidget } from '../widgets';
|
||||||
|
|
||||||
|
const definition = defineWidget({
|
||||||
|
id: 'rss',
|
||||||
|
icon: IconRss,
|
||||||
|
options: {
|
||||||
|
rssFeedUrl: {
|
||||||
|
type: 'text',
|
||||||
|
defaultValue: '',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
gridstack: {
|
||||||
|
minWidth: 2,
|
||||||
|
minHeight: 2,
|
||||||
|
maxWidth: 12,
|
||||||
|
maxHeight: 12,
|
||||||
|
},
|
||||||
|
component: RssTile,
|
||||||
|
});
|
||||||
|
|
||||||
|
export type IRssWidget = IWidget<(typeof definition)['id'], typeof definition>;
|
||||||
|
|
||||||
|
interface RssTileProps {
|
||||||
|
widget: IRssWidget;
|
||||||
|
}
|
||||||
|
|
||||||
|
function RssTile({ widget }: RssTileProps) {
|
||||||
|
const { t } = useTranslation('modules/rss');
|
||||||
|
const { data, isLoading, isFetching, isError, refetch } = useGetRssFeed(
|
||||||
|
widget.properties.rssFeedUrl
|
||||||
|
);
|
||||||
|
const { classes } = useStyles();
|
||||||
|
const [loadingOverlayVisible, setLoadingOverlayVisible] = useState(false);
|
||||||
|
const { ref, height } = useElementSize();
|
||||||
|
|
||||||
|
if (!data || isLoading) {
|
||||||
|
return (
|
||||||
|
<Center>
|
||||||
|
<Loader />
|
||||||
|
</Center>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.success || isError) {
|
||||||
|
return (
|
||||||
|
<Center h="100%">
|
||||||
|
<Stack align="center">
|
||||||
|
<IconRss size={40} strokeWidth={1} />
|
||||||
|
<Title order={6}>{t('card.errors.general.title')}</Title>
|
||||||
|
<Text align="center">{t('card.errors.general.text')}</Text>
|
||||||
|
</Stack>
|
||||||
|
</Center>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack ref={ref} h="100%">
|
||||||
|
<LoadingOverlay visible={loadingOverlayVisible} />
|
||||||
|
<Flex gap="md">
|
||||||
|
{data.feed.image ? (
|
||||||
|
<Image
|
||||||
|
src={data.feed.image.url}
|
||||||
|
alt={data.feed.image.title}
|
||||||
|
width="auto"
|
||||||
|
height={40}
|
||||||
|
mx="auto"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Title order={6}>{data.feed.title}</Title>
|
||||||
|
)}
|
||||||
|
<UnstyledButton
|
||||||
|
onClick={async () => {
|
||||||
|
setLoadingOverlayVisible(true);
|
||||||
|
await Promise.all([sleep(1500), refetch()]);
|
||||||
|
setLoadingOverlayVisible(false);
|
||||||
|
}}
|
||||||
|
disabled={isFetching || isLoading}
|
||||||
|
>
|
||||||
|
<ActionIcon>
|
||||||
|
<IconRefresh />
|
||||||
|
</ActionIcon>
|
||||||
|
</UnstyledButton>
|
||||||
|
</Flex>
|
||||||
|
<ScrollArea className="scroll-area-w100" w="100%">
|
||||||
|
<Stack w="100%" spacing="xs">
|
||||||
|
{data.feed.items.map((item: any, index: number) => (
|
||||||
|
<Card
|
||||||
|
key={index}
|
||||||
|
withBorder
|
||||||
|
component={Link}
|
||||||
|
href={item.link}
|
||||||
|
radius="md"
|
||||||
|
target="_blank"
|
||||||
|
w="100%"
|
||||||
|
>
|
||||||
|
{item.enclosure && (
|
||||||
|
// eslint-disable-next-line @next/next/no-img-element
|
||||||
|
<img
|
||||||
|
className={classes.backgroundImage}
|
||||||
|
src={item.enclosure.url ?? undefined}
|
||||||
|
alt="backdrop"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Flex gap="xs">
|
||||||
|
<MediaQuery query="(max-width: 1200px)" styles={{ display: 'none' }}>
|
||||||
|
<Image
|
||||||
|
src={item.enclosure?.url ?? undefined}
|
||||||
|
width={140}
|
||||||
|
height={140}
|
||||||
|
radius="md"
|
||||||
|
withPlaceholder
|
||||||
|
/>
|
||||||
|
</MediaQuery>
|
||||||
|
<Flex gap={2} direction="column">
|
||||||
|
{item.categories && (
|
||||||
|
<Flex gap="xs" wrap="wrap" h={20} style={{ overflow: 'hidden' }}>
|
||||||
|
{item.categories.map((category: any, categoryIndex: number) => (
|
||||||
|
<Badge key={categoryIndex}>{category._}</Badge>
|
||||||
|
))}
|
||||||
|
</Flex>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Text lineClamp={2}>{item.title}</Text>
|
||||||
|
<Text color="dimmed" size="xs" lineClamp={3}>
|
||||||
|
{item.content}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
{item.pubDate && <TimeDisplay date={item.pubDate} />}
|
||||||
|
</Flex>
|
||||||
|
</Flex>
|
||||||
|
</Card>
|
||||||
|
))}
|
||||||
|
</Stack>
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
<Flex wrap="wrap" columnGap="md">
|
||||||
|
<Group spacing="sm">
|
||||||
|
<IconCopyright size={14} />
|
||||||
|
<Text color="dimmed" size="sm">
|
||||||
|
{data.feed.copyright}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
<Group>
|
||||||
|
<IconCalendarTime size={14} />
|
||||||
|
<Text color="dimmed" size="sm">
|
||||||
|
{data.feed.pubDate}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
<Group>
|
||||||
|
<IconBulldozer size={14} />
|
||||||
|
<Text color="dimmed" size="sm">
|
||||||
|
{data.feed.lastBuildDate}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
{data.feed.feedUrl && (
|
||||||
|
<Group spacing="sm">
|
||||||
|
<IconSpeakerphone size={14} />
|
||||||
|
<Text
|
||||||
|
color="dimmed"
|
||||||
|
size="sm"
|
||||||
|
variant="link"
|
||||||
|
target="_blank"
|
||||||
|
component={Link}
|
||||||
|
href={data.feed.feedUrl}
|
||||||
|
>
|
||||||
|
Feed URL
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const TimeDisplay = ({ date }: { date: string }) => (
|
||||||
|
<Group mt="auto" spacing="xs">
|
||||||
|
<IconClock size={14} />
|
||||||
|
<Text size="xs" color="dimmed">
|
||||||
|
{date}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
|
||||||
|
const useStyles = createStyles(({ colorScheme }) => ({
|
||||||
|
backgroundImage: {
|
||||||
|
position: 'absolute',
|
||||||
|
width: '100%',
|
||||||
|
height: '100%',
|
||||||
|
filter: colorScheme === 'dark' ? 'blur(30px)' : 'blur(15px)',
|
||||||
|
transform: 'scaleX(-1)',
|
||||||
|
opacity: colorScheme === 'dark' ? 0.3 : 0.2,
|
||||||
|
transition: 'ease-in-out 0.2s',
|
||||||
|
|
||||||
|
'&:hover': {
|
||||||
|
opacity: colorScheme === 'dark' ? 0.4 : 0.3,
|
||||||
|
filter: 'blur(40px) brightness(0.7)',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export default definition;
|
||||||
45
yarn.lock
45
yarn.lock
@@ -3751,7 +3751,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"entities@npm:^2.0.0":
|
"entities@npm:^2.0.0, entities@npm:^2.0.3":
|
||||||
version: 2.2.0
|
version: 2.2.0
|
||||||
resolution: "entities@npm:2.2.0"
|
resolution: "entities@npm:2.2.0"
|
||||||
checksum: 19010dacaf0912c895ea262b4f6128574f9ccf8d4b3b65c7e8334ad0079b3706376360e28d8843ff50a78aabcb8f08f0a32dbfacdc77e47ed77ca08b713669b3
|
checksum: 19010dacaf0912c895ea262b4f6128574f9ccf8d4b3b65c7e8334ad0079b3706376360e28d8843ff50a78aabcb8f08f0a32dbfacdc77e47ed77ca08b713669b3
|
||||||
@@ -4976,6 +4976,7 @@ __metadata:
|
|||||||
eslint-plugin-unused-imports: ^2.0.0
|
eslint-plugin-unused-imports: ^2.0.0
|
||||||
fily-publish-gridstack: ^0.0.13
|
fily-publish-gridstack: ^0.0.13
|
||||||
framer-motion: ^6.5.1
|
framer-motion: ^6.5.1
|
||||||
|
html-entities: ^2.3.3
|
||||||
i18next: ^21.9.1
|
i18next: ^21.9.1
|
||||||
jest: ^28.1.3
|
jest: ^28.1.3
|
||||||
js-file-download: ^0.4.12
|
js-file-download: ^0.4.12
|
||||||
@@ -4987,6 +4988,7 @@ __metadata:
|
|||||||
react: ^18.2.0
|
react: ^18.2.0
|
||||||
react-dom: ^18.2.0
|
react-dom: ^18.2.0
|
||||||
react-simple-code-editor: ^0.13.1
|
react-simple-code-editor: ^0.13.1
|
||||||
|
rss-parser: ^3.12.0
|
||||||
sabnzbd-api: ^1.5.0
|
sabnzbd-api: ^1.5.0
|
||||||
sass: ^1.56.1
|
sass: ^1.56.1
|
||||||
turbo: ^1.7.4
|
turbo: ^1.7.4
|
||||||
@@ -5008,6 +5010,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"html-entities@npm:^2.3.3":
|
||||||
|
version: 2.3.3
|
||||||
|
resolution: "html-entities@npm:2.3.3"
|
||||||
|
checksum: 92521501da8aa5f66fee27f0f022d6e9ceae62667dae93aa6a2f636afa71ad530b7fb24a18d4d6c124c9885970cac5f8a52dbf1731741161002816ae43f98196
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"html-escaper@npm:^2.0.0":
|
"html-escaper@npm:^2.0.0":
|
||||||
version: 2.0.2
|
version: 2.0.2
|
||||||
resolution: "html-escaper@npm:2.0.2"
|
resolution: "html-escaper@npm:2.0.2"
|
||||||
@@ -7666,6 +7675,16 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"rss-parser@npm:^3.12.0":
|
||||||
|
version: 3.12.0
|
||||||
|
resolution: "rss-parser@npm:3.12.0"
|
||||||
|
dependencies:
|
||||||
|
entities: ^2.0.3
|
||||||
|
xml2js: ^0.4.19
|
||||||
|
checksum: aa0f0eb2e3a5c70677a1c7cb6c2e96420f12c8963a8bed922ec2ff1bb9dbbb725fc5783be31ca8140154c3d5589ccd31580ced7d32ebd0dda7572f78ce242a41
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"run-parallel@npm:^1.1.9":
|
"run-parallel@npm:^1.1.9":
|
||||||
version: 1.2.0
|
version: 1.2.0
|
||||||
resolution: "run-parallel@npm:1.2.0"
|
resolution: "run-parallel@npm:1.2.0"
|
||||||
@@ -7748,6 +7767,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"sax@npm:>=0.6.0":
|
||||||
|
version: 1.2.4
|
||||||
|
resolution: "sax@npm:1.2.4"
|
||||||
|
checksum: d3df7d32b897a2c2f28e941f732c71ba90e27c24f62ee918bd4d9a8cfb3553f2f81e5493c7f0be94a11c1911b643a9108f231dd6f60df3fa9586b5d2e3e9e1fe
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"scheduler@npm:^0.23.0":
|
"scheduler@npm:^0.23.0":
|
||||||
version: 0.23.0
|
version: 0.23.0
|
||||||
resolution: "scheduler@npm:0.23.0"
|
resolution: "scheduler@npm:0.23.0"
|
||||||
@@ -8842,6 +8868,23 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"xml2js@npm:^0.4.19":
|
||||||
|
version: 0.4.23
|
||||||
|
resolution: "xml2js@npm:0.4.23"
|
||||||
|
dependencies:
|
||||||
|
sax: ">=0.6.0"
|
||||||
|
xmlbuilder: ~11.0.0
|
||||||
|
checksum: ca0cf2dfbf6deeaae878a891c8fbc0db6fd04398087084edf143cdc83d0509ad0fe199b890f62f39c4415cf60268a27a6aed0d343f0658f8779bd7add690fa98
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"xmlbuilder@npm:~11.0.0":
|
||||||
|
version: 11.0.1
|
||||||
|
resolution: "xmlbuilder@npm:11.0.1"
|
||||||
|
checksum: 7152695e16f1a9976658215abab27e55d08b1b97bca901d58b048d2b6e106b5af31efccbdecf9b07af37c8377d8e7e821b494af10b3a68b0ff4ae60331b415b0
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"xtend@npm:^4.0.0":
|
"xtend@npm:^4.0.0":
|
||||||
version: 4.0.2
|
version: 4.0.2
|
||||||
resolution: "xtend@npm:4.0.2"
|
resolution: "xtend@npm:4.0.2"
|
||||||
|
|||||||
Reference in New Issue
Block a user