Progress on #18
This commit is contained in:
parent
a9130c1af8
commit
6a2dc0d4d8
7 changed files with 167 additions and 109 deletions
|
@ -1,8 +1,7 @@
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import styled, { DefaultTheme } from 'styled-components';
|
import styled, { css, DefaultTheme } from 'styled-components';
|
||||||
import { CustomLink } from '../../interfaces/LinkTypes';
|
import { Service, Game } from '../../interfaces/CardTypes';
|
||||||
import { Service } from '../../interfaces/Services';
|
|
||||||
|
|
||||||
// needed for Online Status checks
|
// needed for Online Status checks
|
||||||
// TODO: migrate to shared Status type for Games and Services
|
// TODO: migrate to shared Status type for Games and Services
|
||||||
|
@ -16,7 +15,7 @@ export const PageTitle = styled.h1`
|
||||||
line-height: 1.15;
|
line-height: 1.15;
|
||||||
font-size: 4rem;
|
font-size: 4rem;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
`
|
`;
|
||||||
|
|
||||||
// replaces .description
|
// replaces .description
|
||||||
export const PageDescription = styled.p`
|
export const PageDescription = styled.p`
|
||||||
|
@ -24,7 +23,7 @@ export const PageDescription = styled.p`
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
font-size: 1.5rem;
|
font-size: 1.5rem;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
`
|
`;
|
||||||
|
|
||||||
// replaces .grid
|
// replaces .grid
|
||||||
export const PageContentBox = styled.div`
|
export const PageContentBox = styled.div`
|
||||||
|
@ -33,12 +32,29 @@ export const PageContentBox = styled.div`
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
max-width: 80%;
|
max-width: 80%;
|
||||||
`
|
`;
|
||||||
|
|
||||||
|
const CardStyle = css`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
position: relative;
|
||||||
|
width: 332px;
|
||||||
|
height: 240px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const CardLink = styled(Link)`
|
||||||
|
${CardStyle}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const CardStyleWrap = styled.div`
|
||||||
|
${CardStyle}
|
||||||
|
`;
|
||||||
|
|
||||||
// replaces .card & .contentcard
|
// replaces .card & .contentcard
|
||||||
export const PageCard = styled.div`
|
export const PageCard = styled.div`
|
||||||
margin: 1rem;
|
margin: 1rem;
|
||||||
padding: 1rem;
|
padding: 1.5rem 0.7rem 1.5rem 0.7rem;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: ${({ theme }) => theme.colors.primary};
|
color: ${({ theme }) => theme.colors.primary};
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
@ -46,7 +62,11 @@ export const PageCard = styled.div`
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
border-color: ${({ theme }) => theme.colors.primary};
|
border-color: ${({ theme }) => theme.colors.primary};
|
||||||
transition: color 0.15s ease, border-color 0.15s ease;
|
transition: color 0.15s ease, border-color 0.15s ease;
|
||||||
max-width: 300px;
|
width: 300px;
|
||||||
|
height: 200px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: space-between;
|
||||||
|
|
||||||
h2 {
|
h2 {
|
||||||
margin: 0 0 1rem 0;
|
margin: 0 0 1rem 0;
|
||||||
|
@ -55,15 +75,20 @@ export const PageCard = styled.div`
|
||||||
|
|
||||||
p {
|
p {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: 1.2rem;
|
font-size: 1rem;
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:focus,:active,:hover {
|
${CardStyleWrap}:focus,${CardStyleWrap}:active,${CardStyleWrap}:hover & {
|
||||||
color: ${({ theme }) => theme.colors.secondary};
|
color: ${({ theme }) => theme.colors.secondary};
|
||||||
border-color: ${({ theme }) => theme.colors.secondary};
|
border-color: ${({ theme }) => theme.colors.secondary};
|
||||||
}
|
}
|
||||||
`
|
|
||||||
|
${CardLink}:focus,${CardLink}:active,${CardLink}:hover & {
|
||||||
|
color: ${({ theme }) => theme.colors.secondary};
|
||||||
|
border-color: ${({ theme }) => theme.colors.secondary};
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
// replaces the three status classes
|
// replaces the three status classes
|
||||||
const OnlineStatus = styled.p<OnlinePropType>`
|
const OnlineStatus = styled.p<OnlinePropType>`
|
||||||
|
@ -84,12 +109,31 @@ const OnlineStatus = styled.p<OnlinePropType>`
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}};
|
}};
|
||||||
`
|
padding: 0.2rem;
|
||||||
|
background-color: ${({ theme }) => theme.colors.background};
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: ${({ theme }) => theme.colors.primary};
|
||||||
|
border-radius: 5px;
|
||||||
|
width: min-content;
|
||||||
|
position: absolute;
|
||||||
|
top: 100; right: 50; bottom: 0; left: 50;
|
||||||
|
offset-position: bottom 10px;
|
||||||
|
transition: color 0.15s ease, border-color 0.15s ease;
|
||||||
|
|
||||||
|
${CardStyleWrap}:focus,${CardStyleWrap}:active,${CardStyleWrap}:hover & {
|
||||||
|
border-color: ${({ theme }) => theme.colors.secondary};
|
||||||
|
}
|
||||||
|
|
||||||
|
${CardLink}:focus,${CardLink}:active,${CardLink}:hover & {
|
||||||
|
border-color: ${({ theme }) => theme.colors.secondary};
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
// replaces .cardwarn
|
// replaces .cardwarn
|
||||||
const CardContentWarning = styled.p`
|
const CardContentWarning = styled.p`
|
||||||
color: ${({ theme }) => theme.colors.secondary};
|
color: ${({ theme }) => theme.colors.secondary};
|
||||||
`
|
|
||||||
|
`;
|
||||||
|
|
||||||
// replaces .contentIcon
|
// replaces .contentIcon
|
||||||
const CardContentTitleIcon = styled.div`
|
const CardContentTitleIcon = styled.div`
|
||||||
|
@ -98,7 +142,7 @@ const CardContentTitleIcon = styled.div`
|
||||||
position: relative;
|
position: relative;
|
||||||
aspect-ratio: 1;
|
aspect-ratio: 1;
|
||||||
height: 1.5rem;
|
height: 1.5rem;
|
||||||
`
|
`;
|
||||||
|
|
||||||
// replaces .contentTitle
|
// replaces .contentTitle
|
||||||
const CardContentTitleWrap = styled.div`
|
const CardContentTitleWrap = styled.div`
|
||||||
|
@ -112,9 +156,9 @@ const CardContentTitleWrap = styled.div`
|
||||||
margin: 0;
|
margin: 0;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
`
|
`;
|
||||||
|
|
||||||
const CardContentTitle = ({ content }: { content: Service | CustomLink }) => {
|
const CardContentTitle = ({ content }: { content: Service | Game }) => {
|
||||||
return (
|
return (
|
||||||
<CardContentTitleWrap>
|
<CardContentTitleWrap>
|
||||||
{
|
{
|
||||||
|
@ -130,25 +174,67 @@ const CardContentTitle = ({ content }: { content: Service | CustomLink }) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Card Content Component for Games Page
|
// Card Content Component for Games Page
|
||||||
export const CardContentGame = ({ content }: { content: CustomLink }) => {
|
export const CardContentGame = ({ content }: { content: Game }) => {
|
||||||
return (
|
let ret;
|
||||||
<>
|
if (content.href) {
|
||||||
<CardContentTitle content={content} />
|
ret = (
|
||||||
<p>{content.desc}</p>
|
<CardLink href={content.href}>
|
||||||
<p>{content.ip}</p>
|
<PageCard>
|
||||||
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
<CardContentTitle content={content} />
|
||||||
</>
|
<p>{content.desc}</p>
|
||||||
)
|
<p>{content.ip}</p>
|
||||||
|
</PageCard>
|
||||||
|
{content.status ?
|
||||||
|
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
||||||
|
: <></>
|
||||||
|
}
|
||||||
|
</CardLink>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
ret = (
|
||||||
|
<CardStyleWrap>
|
||||||
|
<PageCard>
|
||||||
|
<CardContentTitle content={content} />
|
||||||
|
<p>{content.desc}</p>
|
||||||
|
<p>{content.ip}</p>
|
||||||
|
</PageCard>
|
||||||
|
{content.status ?
|
||||||
|
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
||||||
|
: <></>
|
||||||
|
}
|
||||||
|
</CardStyleWrap>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Card Content Component for Services Page
|
// Card Content Component for Services Page
|
||||||
export const CardContentService = ({ content }: { content: Service }) => {
|
export const CardContentService = ({ content }: { content: Service }) => {
|
||||||
return (
|
let ret;
|
||||||
<>
|
if (content.href) {
|
||||||
<CardContentTitle content={content} />
|
ret = (
|
||||||
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
<CardLink href={content.href}>
|
||||||
<p>{content.desc}</p>
|
<PageCard>
|
||||||
<CardContentWarning>{content.warn}</CardContentWarning>
|
<CardContentTitle content={content} />
|
||||||
</>
|
<p>{content.desc}</p>
|
||||||
)
|
<CardContentWarning>{content.warn}</CardContentWarning>
|
||||||
|
</PageCard>
|
||||||
|
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
||||||
|
</CardLink>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
ret = (
|
||||||
|
<CardStyleWrap>
|
||||||
|
<PageCard>
|
||||||
|
<CardContentTitle content={content} />
|
||||||
|
<p>{content.desc}</p>
|
||||||
|
<CardContentWarning>{content.warn}</CardContentWarning>
|
||||||
|
</PageCard>
|
||||||
|
<OnlineStatus status={content.status}>{content.status}</OnlineStatus>
|
||||||
|
</CardStyleWrap>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
}
|
}
|
|
@ -31,12 +31,11 @@ export const Footer = styled.footer`
|
||||||
border-top: 1px solid ${({ theme }) => theme.colors.primary };
|
border-top: 1px solid ${({ theme }) => theme.colors.primary };
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
`
|
|
||||||
|
|
||||||
// TODO
|
a {
|
||||||
/* .footer a {
|
display: flex;
|
||||||
display: flex;
|
justify-content: center;
|
||||||
justify-content: center;
|
align-items: center;
|
||||||
align-items: center;
|
flex-grow: 1;
|
||||||
flex-grow: 1;
|
}
|
||||||
} */
|
`
|
||||||
|
|
|
@ -1,3 +1,16 @@
|
||||||
|
export interface EntryList {
|
||||||
|
services: Service[],
|
||||||
|
games: Game[]
|
||||||
|
}
|
||||||
|
export interface Game {
|
||||||
|
name: string,
|
||||||
|
icon: string,
|
||||||
|
href: string,
|
||||||
|
desc: string,
|
||||||
|
ip: string,
|
||||||
|
status: Status,
|
||||||
|
}
|
||||||
|
|
||||||
export interface Service {
|
export interface Service {
|
||||||
name: string,
|
name: string,
|
||||||
icon: string,
|
icon: string,
|
||||||
|
@ -7,10 +20,10 @@ export interface Service {
|
||||||
type: ServiceType,
|
type: ServiceType,
|
||||||
docker_container_name: string,
|
docker_container_name: string,
|
||||||
location: ServiceLocation,
|
location: ServiceLocation,
|
||||||
status: ServiceStatus
|
status: Status,
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ServiceStatus {
|
export enum Status {
|
||||||
online = "Online",
|
online = "Online",
|
||||||
offline = "Offline",
|
offline = "Offline",
|
||||||
loading = "Loading",
|
loading = "Loading",
|
||||||
|
@ -26,3 +39,4 @@ export enum ServiceType {
|
||||||
docker = "docker",
|
docker = "docker",
|
||||||
app = "app"
|
app = "app"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,17 +0,0 @@
|
||||||
import { Service } from "./Services"
|
|
||||||
|
|
||||||
export interface EntryList {
|
|
||||||
services: Service[],
|
|
||||||
games: CustomLink[]
|
|
||||||
}
|
|
||||||
export interface CustomLink {
|
|
||||||
name: string,
|
|
||||||
href: string,
|
|
||||||
desc: string,
|
|
||||||
ip: string,
|
|
||||||
type: string,
|
|
||||||
location: string,
|
|
||||||
status: string,
|
|
||||||
docker_container_name: string
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import fsPromises from 'fs/promises'
|
import fsPromises from 'fs/promises'
|
||||||
import path from 'path'
|
import path from 'path'
|
||||||
import { Service, ServiceStatus } from '../../interfaces/Services';
|
import { Service, Status } from '../../interfaces/CardTypes';
|
||||||
|
|
||||||
export default async function ServicesAPI(req: any, res: any) {
|
export default async function ServicesAPI(req: any, res: any) {
|
||||||
try {
|
try {
|
||||||
|
@ -8,7 +8,7 @@ export default async function ServicesAPI(req: any, res: any) {
|
||||||
const data = await fsPromises.readFile(filePath)
|
const data = await fsPromises.readFile(filePath)
|
||||||
.then((file) => JSON.parse(file.toString()));
|
.then((file) => JSON.parse(file.toString()));
|
||||||
data.services.forEach((service: Service) => {
|
data.services.forEach((service: Service) => {
|
||||||
service.status = ServiceStatus.loading;
|
service.status = Status.loading;
|
||||||
});
|
});
|
||||||
|
|
||||||
res.status(200).json(data.services);
|
res.status(200).json(data.services);
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import Head from 'next/head'
|
import Head from 'next/head'
|
||||||
import fsPromises from 'fs/promises'
|
import fsPromises from 'fs/promises'
|
||||||
import path from 'path'
|
import path from 'path'
|
||||||
import type { CustomLink, EntryList } from '../interfaces/LinkTypes'
|
import { EntryList, Game } from '../interfaces/CardTypes';
|
||||||
import { PageContentBox, PageCard, PageDescription, PageTitle, CardContentGame } from '../components/styles/content'
|
import { PageContentBox, PageDescription, PageTitle, CardContentGame } from '../components/styles/content'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
|
|
||||||
function Servers(props: EntryList) {
|
function Servers(props: EntryList) {
|
||||||
|
@ -24,25 +24,9 @@ function Servers(props: EntryList) {
|
||||||
</PageDescription>
|
</PageDescription>
|
||||||
|
|
||||||
<PageContentBox>
|
<PageContentBox>
|
||||||
{Object.values(serverList).map((item: CustomLink) => {
|
{Object.values(serverList).map((item: Game) => (
|
||||||
if (item.href != null) {
|
<CardContentGame key={item.name} content={item} />
|
||||||
return (
|
))}
|
||||||
<PageCard key={item.name}>
|
|
||||||
<Link href={item.href}>
|
|
||||||
<CardContentGame content={item} />
|
|
||||||
</Link>
|
|
||||||
</PageCard>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return (
|
|
||||||
<PageCard key={item.name}>
|
|
||||||
<CardContentGame content={item} />
|
|
||||||
</PageCard>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)}
|
|
||||||
</PageContentBox>
|
</PageContentBox>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import Head from 'next/head'
|
import Head from 'next/head'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
import { Service, ServiceStatus, ServiceType, ServiceLocation } from '../interfaces/Services';
|
import { Service, Status, ServiceType, ServiceLocation } from '../interfaces/CardTypes';
|
||||||
import Dockerode from 'dockerode';
|
import Dockerode from 'dockerode';
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import useSWR from 'swr';
|
import useSWR from 'swr';
|
||||||
|
@ -19,11 +19,7 @@ function Services() {
|
||||||
content =
|
content =
|
||||||
<PageContentBox>
|
<PageContentBox>
|
||||||
{initialData?.map((item: Service) => (
|
{initialData?.map((item: Service) => (
|
||||||
<PageCard key={item.name}>
|
<CardContentService key={item.name} content={item} />
|
||||||
<Link key={item.name} href={item.href}>
|
|
||||||
<CardContentService content={item} />
|
|
||||||
</Link>
|
|
||||||
</PageCard>
|
|
||||||
))}
|
))}
|
||||||
</PageContentBox>
|
</PageContentBox>
|
||||||
|
|
||||||
|
@ -32,11 +28,7 @@ function Services() {
|
||||||
content =
|
content =
|
||||||
<PageContentBox>
|
<PageContentBox>
|
||||||
{fullData.map((item: Service) => (
|
{fullData.map((item: Service) => (
|
||||||
<PageCard key={item.name}>
|
<CardContentService key={item.name} content={item} />
|
||||||
<Link key={item.name} href={item.href}>
|
|
||||||
<CardContentService content={item} />
|
|
||||||
</Link>
|
|
||||||
</PageCard>
|
|
||||||
))}
|
))}
|
||||||
</PageContentBox>
|
</PageContentBox>
|
||||||
}
|
}
|
||||||
|
@ -81,19 +73,19 @@ async function getStatus(entry: Service, containers: Dockerode.ContainerInfo[])
|
||||||
case 200:
|
case 200:
|
||||||
case 301:
|
case 301:
|
||||||
case 302:
|
case 302:
|
||||||
entry.status = ServiceStatus.online;
|
entry.status = Status.online;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Error pinging Website: ", error);
|
console.error("Error pinging Website: ", error);
|
||||||
entry.status = ServiceStatus.error;
|
entry.status = Status.error;
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
// Type Docker
|
// Type Docker
|
||||||
|
@ -107,11 +99,11 @@ async function getStatus(entry: Service, containers: Dockerode.ContainerInfo[])
|
||||||
// so far only "running" is properly implemented, mroe cases to follow as needed
|
// so far only "running" is properly implemented, mroe cases to follow as needed
|
||||||
switch (container.State) {
|
switch (container.State) {
|
||||||
case "running":
|
case "running":
|
||||||
entry.status = ServiceStatus.online;
|
entry.status = Status.online;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
console.log("Container Status " + container.State + " has no case implemented");
|
console.log("Container Status " + container.State + " has no case implemented");
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
found = true;
|
found = true;
|
||||||
// cancel the for
|
// cancel the for
|
||||||
|
@ -119,7 +111,7 @@ async function getStatus(entry: Service, containers: Dockerode.ContainerInfo[])
|
||||||
}
|
}
|
||||||
// If container name is not missing the container is set to offline
|
// If container name is not missing the container is set to offline
|
||||||
else {
|
else {
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!found) {
|
if (!found) {
|
||||||
|
@ -129,13 +121,13 @@ async function getStatus(entry: Service, containers: Dockerode.ContainerInfo[])
|
||||||
// if name is null do not enter for loop
|
// if name is null do not enter for loop
|
||||||
else {
|
else {
|
||||||
console.error("Container Name not specified");
|
console.error("Container Name not specified");
|
||||||
entry.status = ServiceStatus.error;
|
entry.status = Status.error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If no Type matches
|
// If no Type matches
|
||||||
else {
|
else {
|
||||||
console.warn("Service Type for Service " + entry.name + " not specified or invalid");
|
console.warn("Service Type for Service " + entry.name + " not specified or invalid");
|
||||||
entry.status = ServiceStatus.error;
|
entry.status = Status.error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Location Other
|
// Location Other
|
||||||
|
@ -149,25 +141,25 @@ async function getStatus(entry: Service, containers: Dockerode.ContainerInfo[])
|
||||||
case 200:
|
case 200:
|
||||||
case 301:
|
case 301:
|
||||||
case 302:
|
case 302:
|
||||||
entry.status = ServiceStatus.online;
|
entry.status = Status.online;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
entry.status = ServiceStatus.offline;
|
entry.status = Status.offline;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error("Error pinging Website: ", error);
|
console.error("Error pinging Website: ", error);
|
||||||
entry.status = ServiceStatus.error;
|
entry.status = Status.error;
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
// If no Location matches
|
// If no Location matches
|
||||||
else {
|
else {
|
||||||
console.warn("Service Location for Service " + entry.name + " not specified");
|
console.warn("Service Location for Service " + entry.name + " not specified");
|
||||||
entry.status = ServiceStatus.error;
|
entry.status = Status.error;
|
||||||
}
|
}
|
||||||
return entry;
|
return entry;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue