mirror of
https://github.com/sergi0g/cup.git
synced 2025-11-08 05:03:49 -05:00
Compare commits
2 Commits
c8229d7370
...
8a5b0555f7
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
8a5b0555f7 | ||
|
|
2e1b0945e0 |
2
Cargo.lock
generated
2
Cargo.lock
generated
@@ -355,7 +355,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cup"
|
name = "cup"
|
||||||
version = "3.2.3"
|
version = "3.3.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bollard",
|
"bollard",
|
||||||
"chrono",
|
"chrono",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cup"
|
name = "cup"
|
||||||
version = "3.2.3"
|
version = "3.3.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import DataLoadingError from "./components/DataLoadingError";
|
|||||||
import Filters from "./components/Filters";
|
import Filters from "./components/Filters";
|
||||||
import { Filter, FilterX } from "lucide-react";
|
import { Filter, FilterX } from "lucide-react";
|
||||||
import { WithTooltip } from "./components/ui/Tooltip";
|
import { WithTooltip } from "./components/ui/Tooltip";
|
||||||
|
import { getDescription } from "./utils";
|
||||||
|
|
||||||
const SORT_ORDER = [
|
const SORT_ORDER = [
|
||||||
"monitored_images",
|
"monitored_images",
|
||||||
@@ -32,6 +33,8 @@ function App() {
|
|||||||
const [showFilters, setShowFilters] = useState<boolean>(false);
|
const [showFilters, setShowFilters] = useState<boolean>(false);
|
||||||
const [filters, setFilters] = useState<FiltersType>({
|
const [filters, setFilters] = useState<FiltersType>({
|
||||||
onlyInUse: false,
|
onlyInUse: false,
|
||||||
|
registries: [],
|
||||||
|
statuses: [],
|
||||||
});
|
});
|
||||||
const [searchQuery, setSearchQuery] = useState("");
|
const [searchQuery, setSearchQuery] = useState("");
|
||||||
|
|
||||||
@@ -39,7 +42,7 @@ function App() {
|
|||||||
if (isError || !data) return <DataLoadingError />;
|
if (isError || !data) return <DataLoadingError />;
|
||||||
const toggleShowFilters = () => {
|
const toggleShowFilters = () => {
|
||||||
if (showFilters) {
|
if (showFilters) {
|
||||||
setFilters({ onlyInUse: false });
|
setFilters({ onlyInUse: false, registries: [], statuses: [] });
|
||||||
}
|
}
|
||||||
setShowFilters(!showFilters);
|
setShowFilters(!showFilters);
|
||||||
};
|
};
|
||||||
@@ -95,7 +98,13 @@ function App() {
|
|||||||
<Search onChange={setSearchQuery} />
|
<Search onChange={setSearchQuery} />
|
||||||
</div>
|
</div>
|
||||||
{showFilters && (
|
{showFilters && (
|
||||||
<Filters filters={filters} setFilters={setFilters} />
|
<Filters
|
||||||
|
filters={filters}
|
||||||
|
setFilters={setFilters}
|
||||||
|
registries={[
|
||||||
|
...new Set(data.images.map((image) => image.parts.registry)),
|
||||||
|
]}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
<ul>
|
<ul>
|
||||||
{Object.entries(
|
{Object.entries(
|
||||||
@@ -116,6 +125,16 @@ function App() {
|
|||||||
.filter((image) =>
|
.filter((image) =>
|
||||||
filters.onlyInUse ? !!image.in_use : true,
|
filters.onlyInUse ? !!image.in_use : true,
|
||||||
)
|
)
|
||||||
|
.filter((image) =>
|
||||||
|
filters.registries.length == 0
|
||||||
|
? true
|
||||||
|
: filters.registries.includes(image.parts.registry),
|
||||||
|
)
|
||||||
|
.filter((image) =>
|
||||||
|
filters.statuses.length == 0
|
||||||
|
? true
|
||||||
|
: filters.statuses.includes(getDescription(image)),
|
||||||
|
)
|
||||||
.filter((image) => image.reference.includes(searchQuery))
|
.filter((image) => image.reference.includes(searchQuery))
|
||||||
.map((image) => (
|
.map((image) => (
|
||||||
<Image data={image} key={image.reference} />
|
<Image data={image} key={image.reference} />
|
||||||
|
|||||||
@@ -1,33 +1,81 @@
|
|||||||
|
import { useState } from "react";
|
||||||
import { theme } from "../theme";
|
import { theme } from "../theme";
|
||||||
import { Filters as FiltersType } from "../types";
|
import { Filters as FiltersType } from "../types";
|
||||||
import { Checkbox } from "./ui/Checkbox";
|
import { Checkbox } from "./ui/Checkbox";
|
||||||
|
import Select from "./ui/Select";
|
||||||
|
import { Server } from "lucide-react";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
filters: FiltersType;
|
filters: FiltersType;
|
||||||
setFilters: (filters: FiltersType) => void;
|
setFilters: (filters: FiltersType) => void;
|
||||||
|
registries: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Filters({ filters, setFilters }: Props) {
|
const STATUSES = [
|
||||||
|
"Major update",
|
||||||
|
"Minor update",
|
||||||
|
"Patch update",
|
||||||
|
"Digest update",
|
||||||
|
"Up to date",
|
||||||
|
"Unknown",
|
||||||
|
];
|
||||||
|
|
||||||
|
export default function Filters({ filters, setFilters, registries }: Props) {
|
||||||
|
const [selectedRegistries, setSelectedRegistries] = useState<
|
||||||
|
FiltersType["registries"]
|
||||||
|
>([]);
|
||||||
|
const [selectedStatuses, setSelectedStatuses] = useState<
|
||||||
|
FiltersType["statuses"]
|
||||||
|
>([]);
|
||||||
|
const handleSelectRegistries = (registries: string[]) => {
|
||||||
|
setSelectedRegistries(registries);
|
||||||
|
setFilters({
|
||||||
|
...filters,
|
||||||
|
registries,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
const handleSelectStatuses = (statuses: string[]) => {
|
||||||
|
if (statuses.every((status) => STATUSES.includes(status))) {
|
||||||
|
setSelectedStatuses(statuses as FiltersType["statuses"]);
|
||||||
|
setFilters({
|
||||||
|
...filters,
|
||||||
|
statuses: statuses as FiltersType["statuses"],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
return (
|
return (
|
||||||
<div className="flex w-fit flex-col gap-2 px-6 py-4">
|
<div className="flex w-full flex-col gap-4 px-6 py-4 sm:flex-row">
|
||||||
<div className="ml-auto flex items-center space-x-2">
|
<div className="flex items-center space-x-2">
|
||||||
<Checkbox
|
<Checkbox
|
||||||
id="inUse"
|
id="inUse"
|
||||||
checked={filters.onlyInUse}
|
checked={filters.onlyInUse}
|
||||||
onCheckedChange={(value) => {
|
onCheckedChange={(value) => {
|
||||||
setFilters({
|
setFilters({
|
||||||
...filters,
|
...filters,
|
||||||
onlyInUse: value === "indeterminate" ? false : value,
|
onlyInUse: value === "indeterminate" ? false : value,
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<label
|
<label
|
||||||
htmlFor="inUse"
|
htmlFor="inUse"
|
||||||
className={`text-sm font-medium leading-none text-${theme}-600 dark:text-${theme}-400 hover:text-black dark:hover:text-white peer-hover:text-black peer-hover:dark:text-white peer-data-[state=checked]:text-black dark:peer-data-[state=checked]:text-white transition-colors duration-200`}
|
className={`text-sm font-medium leading-none text-${theme}-600 dark:text-${theme}-400 transition-colors duration-200 hover:text-black peer-hover:text-black peer-data-[state=checked]:text-black dark:hover:text-white peer-hover:dark:text-white dark:peer-data-[state=checked]:text-white`}
|
||||||
>
|
>
|
||||||
Hide unused images
|
Hide unused images
|
||||||
</label>
|
</label>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<Select
|
||||||
|
Icon={Server}
|
||||||
|
items={registries}
|
||||||
|
placeholder="Registry"
|
||||||
|
selectedItems={selectedRegistries}
|
||||||
|
setSelectedItems={handleSelectRegistries}
|
||||||
|
/>
|
||||||
|
<Select
|
||||||
|
items={STATUSES}
|
||||||
|
placeholder="Update type"
|
||||||
|
selectedItems={selectedStatuses}
|
||||||
|
setSelectedItems={handleSelectStatuses}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import {
|
|||||||
X,
|
X,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
import Badge from "./Badge";
|
import Badge from "./Badge";
|
||||||
|
import { getDescription } from "../utils";
|
||||||
|
|
||||||
const clickable_registries = [
|
const clickable_registries = [
|
||||||
"registry-1.docker.io",
|
"registry-1.docker.io",
|
||||||
@@ -39,7 +40,7 @@ export default function Image({ data }: { data: Image }) {
|
|||||||
data.result.info?.type == "version"
|
data.result.info?.type == "version"
|
||||||
? data.reference.split(":")[0] + ":" + data.result.info.new_tag
|
? data.reference.split(":")[0] + ":" + data.result.info.new_tag
|
||||||
: data.reference;
|
: data.reference;
|
||||||
const info = getInfo(data)!;
|
const info = getInfo(data);
|
||||||
let url: string | null = null;
|
let url: string | null = null;
|
||||||
if (data.url) {
|
if (data.url) {
|
||||||
url = data.url;
|
url = data.url;
|
||||||
@@ -182,54 +183,49 @@ export default function Image({ data }: { data: Image }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getInfo(data: Image):
|
function getInfo(data: Image): {
|
||||||
| {
|
color: string;
|
||||||
color: string;
|
icon: typeof HelpCircle;
|
||||||
icon: typeof HelpCircle;
|
description: string;
|
||||||
description: string;
|
} {
|
||||||
}
|
const description = getDescription(data);
|
||||||
| undefined {
|
switch (description) {
|
||||||
switch (data.result.has_update) {
|
case "Unknown":
|
||||||
case null:
|
|
||||||
return {
|
return {
|
||||||
color: "text-gray-500",
|
color: "text-gray-500",
|
||||||
icon: HelpCircle,
|
icon: HelpCircle,
|
||||||
description: "Unknown",
|
description,
|
||||||
};
|
};
|
||||||
case false:
|
case "Up to date":
|
||||||
return {
|
return {
|
||||||
color: "text-green-500",
|
color: "text-green-500",
|
||||||
icon: CircleCheck,
|
icon: CircleCheck,
|
||||||
description: "Up to date",
|
description,
|
||||||
|
};
|
||||||
|
|
||||||
|
case "Major update":
|
||||||
|
return {
|
||||||
|
color: "text-red-500",
|
||||||
|
icon: CircleArrowUp,
|
||||||
|
description,
|
||||||
|
};
|
||||||
|
case "Minor update":
|
||||||
|
return {
|
||||||
|
color: "text-yellow-500",
|
||||||
|
icon: CircleArrowUp,
|
||||||
|
description,
|
||||||
|
};
|
||||||
|
case "Patch update":
|
||||||
|
return {
|
||||||
|
color: "text-blue-500",
|
||||||
|
icon: CircleArrowUp,
|
||||||
|
description,
|
||||||
|
};
|
||||||
|
case "Digest update":
|
||||||
|
return {
|
||||||
|
color: "text-blue-500",
|
||||||
|
icon: CircleArrowUp,
|
||||||
|
description,
|
||||||
};
|
};
|
||||||
case true:
|
|
||||||
if (data.result.info?.type === "version") {
|
|
||||||
switch (data.result.info.version_update_type) {
|
|
||||||
case "major":
|
|
||||||
return {
|
|
||||||
color: "text-red-500",
|
|
||||||
icon: CircleArrowUp,
|
|
||||||
description: "Major update",
|
|
||||||
};
|
|
||||||
case "minor":
|
|
||||||
return {
|
|
||||||
color: "text-yellow-500",
|
|
||||||
icon: CircleArrowUp,
|
|
||||||
description: "Minor update",
|
|
||||||
};
|
|
||||||
case "patch":
|
|
||||||
return {
|
|
||||||
color: "text-blue-500",
|
|
||||||
icon: CircleArrowUp,
|
|
||||||
description: "Patch update",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
} else if (data.result.info?.type === "digest") {
|
|
||||||
return {
|
|
||||||
color: "text-blue-500",
|
|
||||||
icon: CircleArrowUp,
|
|
||||||
description: "Update available",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
84
web/src/components/ui/Select.tsx
Normal file
84
web/src/components/ui/Select.tsx
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
import {
|
||||||
|
Listbox,
|
||||||
|
ListboxButton,
|
||||||
|
ListboxOptions,
|
||||||
|
ListboxOption,
|
||||||
|
} from "@headlessui/react";
|
||||||
|
import { ChevronDown, Check } from "lucide-react";
|
||||||
|
import { theme } from "../../theme";
|
||||||
|
import { cn } from "../../utils";
|
||||||
|
import { Server } from "lucide-react";
|
||||||
|
|
||||||
|
export default function Select({
|
||||||
|
items,
|
||||||
|
Icon,
|
||||||
|
placeholder,
|
||||||
|
selectedItems,
|
||||||
|
setSelectedItems,
|
||||||
|
}: {
|
||||||
|
items: string[];
|
||||||
|
Icon?: typeof Server;
|
||||||
|
placeholder: string;
|
||||||
|
selectedItems: string[];
|
||||||
|
setSelectedItems: (items: string[]) => void;
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<Listbox value={selectedItems} onChange={setSelectedItems} multiple>
|
||||||
|
<div className="relative">
|
||||||
|
<ListboxButton
|
||||||
|
className={cn(
|
||||||
|
`flex overflow-x-hidden w-full gap-2 rounded-md bg-${theme}-100 dark:bg-${theme}-900 border border-${theme}-200 dark:border-${theme}-700 group relative items-center py-1.5 pl-3 pr-2 text-left transition-colors duration-200 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-1 focus-visible:outline-blue-500 sm:text-sm/6`,
|
||||||
|
selectedItems.length == 0
|
||||||
|
? `text-${theme}-600 dark:text-${theme}-400 hover:text-black hover:dark:text-white`
|
||||||
|
: "text-black dark:text-white",
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{Icon && (
|
||||||
|
<Icon
|
||||||
|
className={cn(
|
||||||
|
"size-4 shrink-0",
|
||||||
|
selectedItems.length == 0
|
||||||
|
? `text-${theme}-600 dark:text-${theme}-400 hover:text-black hover:dark:text-white`
|
||||||
|
: "text-black dark:text-white",
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<span className="truncate">
|
||||||
|
{selectedItems.length == 0
|
||||||
|
? placeholder
|
||||||
|
: selectedItems.length == 1
|
||||||
|
? selectedItems[0]
|
||||||
|
: `${selectedItems[0]} +${(selectedItems.length - 1).toString()} more`}</span>
|
||||||
|
|
||||||
|
<ChevronDown
|
||||||
|
aria-hidden="true"
|
||||||
|
className={`size-5 shrink-0 ml-auto self-center text-${theme}-600 dark:text-${theme}-400 transition-colors duration-200 group-hover:text-black sm:size-4 group-hover:dark:text-white`}
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="absolute -bottom-px left-1/2 h-full w-0 -translate-x-1/2 rounded-md border-b-2 border-b-blue-600 transition-all duration-200 group-data-[open]:w-[calc(100%+2px)]"
|
||||||
|
style={{ clipPath: "inset(calc(100% - 2px) 0 0 0)" }}
|
||||||
|
></div>
|
||||||
|
</ListboxButton>
|
||||||
|
<ListboxOptions
|
||||||
|
transition
|
||||||
|
className={`absolute z-10 mt-1 max-h-56 w-max overflow-y-auto overflow-x-hidden rounded-md bg-${theme}-100 dark:bg-${theme}-900 border border-${theme}-200 dark:border-${theme}-700 text-base shadow-lg ring-1 ring-black/5 focus:outline-none data-[closed]:data-[leave]:opacity-0 data-[leave]:transition data-[leave]:duration-100 data-[leave]:ease-in sm:text-sm`}
|
||||||
|
>
|
||||||
|
{items.map((item) => (
|
||||||
|
<ListboxOption
|
||||||
|
key={item}
|
||||||
|
value={item}
|
||||||
|
className={`group relative cursor-pointer text-nowrap py-2 pl-3 pr-9 data-[focus]:outline-none text-${theme}-600 dark:text-${theme}-400 transition-colors duration-200 data-[focus]:bg-black/10 data-[focus]:text-black dark:data-[focus]:bg-white/10 data-[focus]:dark:text-white`}
|
||||||
|
>
|
||||||
|
{item}
|
||||||
|
<span
|
||||||
|
className={`absolute inset-y-0 right-2 flex items-center text-${theme}-600 dark:text-${theme}-400 group-[:not([data-selected])]:hidden group-data-[focus]:text-black group-data-[focus]:dark:text-white`}
|
||||||
|
>
|
||||||
|
<Check aria-hidden="true" className="size-4" />
|
||||||
|
</span>
|
||||||
|
</ListboxOption>
|
||||||
|
))}
|
||||||
|
</ListboxOptions>
|
||||||
|
</div>
|
||||||
|
</Listbox>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -47,4 +47,13 @@ interface DigestInfo {
|
|||||||
|
|
||||||
export interface Filters {
|
export interface Filters {
|
||||||
onlyInUse: boolean;
|
onlyInUse: boolean;
|
||||||
|
registries: string[];
|
||||||
|
statuses: (
|
||||||
|
| "Major update"
|
||||||
|
| "Minor update"
|
||||||
|
| "Patch update"
|
||||||
|
| "Digest update"
|
||||||
|
| "Up to date"
|
||||||
|
| "Unknown"
|
||||||
|
)[];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,30 @@
|
|||||||
import { clsx, type ClassValue } from "clsx";
|
import { clsx, type ClassValue } from "clsx";
|
||||||
import { twMerge } from "tailwind-merge";
|
import { twMerge } from "tailwind-merge";
|
||||||
|
import type { Image } from "./types";
|
||||||
|
|
||||||
export function cn(...inputs: ClassValue[]) {
|
export function cn(...inputs: ClassValue[]) {
|
||||||
return twMerge(clsx(inputs));
|
return twMerge(clsx(inputs));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getDescription(image: Image) {
|
||||||
|
switch (image.result.has_update) {
|
||||||
|
case null:
|
||||||
|
return "Unknown";
|
||||||
|
case false:
|
||||||
|
return "Up to date";
|
||||||
|
case true:
|
||||||
|
if (image.result.info?.type === "version") {
|
||||||
|
switch (image.result.info.version_update_type) {
|
||||||
|
case "major":
|
||||||
|
return "Major update";
|
||||||
|
case "minor":
|
||||||
|
return "Minor update";
|
||||||
|
case "patch":
|
||||||
|
return "Patch update";
|
||||||
|
}
|
||||||
|
} else if (image.result.info?.type === "digest") {
|
||||||
|
return "Digest update";
|
||||||
|
}
|
||||||
|
return "Unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -32,14 +32,14 @@ export default {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
pattern: /text-(gray|neutral)-600/,
|
pattern: /text-(gray|neutral)-600/,
|
||||||
variants: ["*", "dark", "hover", "placeholder"],
|
variants: ["*", "dark", "hover", "placeholder", "data-[placeholder]"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pattern: /text-(gray|neutral)-400/,
|
pattern: /text-(gray|neutral)-400/,
|
||||||
variants: ["*:dark", "dark", "dark:hover", "placeholder:dark"],
|
variants: ["*:dark", "dark", "dark:hover", "placeholder:dark", "data-[placeholder]:dark"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pattern: /text-(gray|neutral)-700/,
|
pattern: /text-(gray|neutral)-(500|700)/,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pattern: /text-(gray|neutral)-950/,
|
pattern: /text-(gray|neutral)-950/,
|
||||||
|
|||||||
Reference in New Issue
Block a user