Compare commits

...

2 Commits

10 changed files with 73 additions and 48 deletions

View File

@ -31,7 +31,7 @@ If you install Garage using Docker, you can install this web UI alongside Garage
```yml ```yml
services: services:
garage: garage:
image: dxflrs/garage:v1.0.0 image: dxflrs/garage:v1.0.1
container_name: garage container_name: garage
volumes: volumes:
- ./garage.toml:/etc/garage.toml - ./garage.toml:/etc/garage.toml
@ -55,7 +55,7 @@ services:
Get the latest binary from the [release page](https://github.com/khairul169/garage-webui/releases/latest) according to your OS architecture. For example: Get the latest binary from the [release page](https://github.com/khairul169/garage-webui/releases/latest) according to your OS architecture. For example:
```sh ```sh
$ wget -O garage-webui https://github.com/khairul169/garage-webui/releases/download/1.0.3/garage-webui-v1.0.3-linux-amd64 $ wget -O garage-webui https://github.com/khairul169/garage-webui/releases/download/1.0.4/garage-webui-v1.0.4-linux-amd64
$ chmod +x garage-webui $ chmod +x garage-webui
$ sudo cp garage-webui /usr/local/bin $ sudo cp garage-webui /usr/local/bin
``` ```

View File

@ -34,13 +34,14 @@ func (b *Buckets) GetAll(w http.ResponseWriter, r *http.Request) {
return return
} }
var bucket schema.Bucket var data schema.Bucket
if err := json.Unmarshal(body, &bucket); err != nil { if err := json.Unmarshal(body, &data); err != nil {
ch <- schema.Bucket{ID: bucket.ID, GlobalAliases: bucket.GlobalAliases} ch <- schema.Bucket{ID: bucket.ID, GlobalAliases: bucket.GlobalAliases}
return return
} }
ch <- bucket data.LocalAliases = bucket.LocalAliases
ch <- data
}() }()
} }

View File

@ -3,12 +3,13 @@ package schema
type GetBucketsRes struct { type GetBucketsRes struct {
ID string `json:"id"` ID string `json:"id"`
GlobalAliases []string `json:"globalAliases"` GlobalAliases []string `json:"globalAliases"`
LocalAliases []string `json:"localAliases"` LocalAliases []LocalAlias `json:"localAliases"`
} }
type Bucket struct { type Bucket struct {
ID string `json:"id"` ID string `json:"id"`
GlobalAliases []string `json:"globalAliases"` GlobalAliases []string `json:"globalAliases"`
LocalAliases []LocalAlias `json:"localAliases"`
WebsiteAccess bool `json:"websiteAccess"` WebsiteAccess bool `json:"websiteAccess"`
WebsiteConfig WebsiteConfig `json:"websiteConfig"` WebsiteConfig WebsiteConfig `json:"websiteConfig"`
Keys []KeyElement `json:"keys"` Keys []KeyElement `json:"keys"`
@ -21,11 +22,16 @@ type Bucket struct {
Quotas Quotas `json:"quotas"` Quotas Quotas `json:"quotas"`
} }
type LocalAlias struct {
AccessKeyID string `json:"accessKeyId"`
Alias string `json:"alias"`
}
type KeyElement struct { type KeyElement struct {
AccessKeyID string `json:"accessKeyId"` AccessKeyID string `json:"accessKeyId"`
Name string `json:"name"` Name string `json:"name"`
Permissions Permissions `json:"permissions"` Permissions Permissions `json:"permissions"`
BucketLocalAliases []interface{} `json:"bucketLocalAliases"` BucketLocalAliases []string `json:"bucketLocalAliases"`
SecretAccessKey string `json:"secretAccessKey"` SecretAccessKey string `json:"secretAccessKey"`
} }

View File

@ -1,12 +1,6 @@
package schema package schema
type Config struct { type Config struct {
CompressionLevel int64 `json:"compression_level" toml:"compression_level"`
DataDir string `json:"data_dir" toml:"data_dir"`
DBEngine string `json:"db_engine" toml:"db_engine"`
MetadataAutoSnapshotInterval string `json:"metadata_auto_snapshot_interval" toml:"metadata_auto_snapshot_interval"`
MetadataDir string `json:"metadata_dir" toml:"metadata_dir"`
ReplicationFactor int64 `json:"replication_factor" toml:"replication_factor"`
RPCBindAddr string `json:"rpc_bind_addr" toml:"rpc_bind_addr"` RPCBindAddr string `json:"rpc_bind_addr" toml:"rpc_bind_addr"`
RPCPublicAddr string `json:"rpc_public_addr" toml:"rpc_public_addr"` RPCPublicAddr string `json:"rpc_public_addr" toml:"rpc_public_addr"`
RPCSecret string `json:"rpc_secret" toml:"rpc_secret"` RPCSecret string `json:"rpc_secret" toml:"rpc_secret"`

View File

@ -1,7 +1,7 @@
{ {
"name": "garage-webui", "name": "garage-webui",
"private": true, "private": true,
"version": "1.0.3", "version": "1.0.4",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev:client": "vite", "dev:client": "vite",

View File

@ -4,7 +4,7 @@ import { readableBytes } from "@/lib/utils";
import Button from "@/components/ui/button"; import Button from "@/components/ui/button";
type Props = { type Props = {
data: Bucket; data: Bucket & { aliases: string[] };
}; };
const BucketCard = ({ data }: Props) => { const BucketCard = ({ data }: Props) => {
@ -15,7 +15,7 @@ const BucketCard = ({ data }: Props) => {
<ArchiveIcon size={28} className="shrink-0" /> <ArchiveIcon size={28} className="shrink-0" />
<p className="text-xl font-medium truncate"> <p className="text-xl font-medium truncate">
{data.globalAliases?.join(", ")} {data.aliases?.join(", ")}
</p> </p>
</div> </div>

View File

@ -84,7 +84,7 @@ const AllowKeyDialog = ({ currentKeys }: Props) => {
Allow Key Allow Key
</Button> </Button>
<Modal ref={dialogRef} backdrop open={isOpen}> <Modal ref={dialogRef} backdrop open={isOpen} className="max-w-2xl">
<Modal.Header className="mb-1">Allow Key</Modal.Header> <Modal.Header className="mb-1">Allow Key</Modal.Header>
<Modal.Body> <Modal.Body>
<p>Enter the key you want to allow access to.</p> <p>Enter the key you want to allow access to.</p>
@ -100,6 +100,7 @@ const AllowKeyDialog = ({ currentKeys }: Props) => {
/> />
Key Key
</label> </label>
<label>Local Aliases</label>
<label className="flex items-center gap-2 cursor-pointer"> <label className="flex items-center gap-2 cursor-pointer">
<Checkbox <Checkbox
color="primary" color="primary"
@ -129,23 +130,31 @@ const AllowKeyDialog = ({ currentKeys }: Props) => {
<Table.Body> <Table.Body>
{!keyFields.length ? ( {!keyFields.length ? (
<tr> <tr>
<td colSpan={4} className="text-center"> <td colSpan={5} className="text-center">
No keys found No keys found
</td> </td>
</tr> </tr>
) : null} ) : null}
{keyFields.map((field, index) => ( {keyFields.map((field, index) => {
const curKey = bucket.keys.find(
(key) => key.accessKeyId === field.keyId
);
return (
<Table.Row key={field.id}> <Table.Row key={field.id}>
<CheckboxField <CheckboxField
form={form} form={form}
name={`keys.${index}.checked`} name={`keys.${index}.checked`}
label={field.name || field.keyId?.substring(0, 8)} label={field.name || field.keyId?.substring(0, 8)}
/> />
<span>
{curKey?.bucketLocalAliases?.join(", ") || "-"}
</span>
<CheckboxField form={form} name={`keys.${index}.read`} /> <CheckboxField form={form} name={`keys.${index}.read`} />
<CheckboxField form={form} name={`keys.${index}.write`} /> <CheckboxField form={form} name={`keys.${index}.write`} />
<CheckboxField form={form} name={`keys.${index}.owner`} /> <CheckboxField form={form} name={`keys.${index}.owner`} />
</Table.Row> </Table.Row>
))} );
})}
</Table.Body> </Table.Body>
</Table> </Table>
</div> </div>

View File

@ -50,6 +50,7 @@ const PermissionsTab = () => {
<Table.Head> <Table.Head>
<span>#</span> <span>#</span>
<span>Key</span> <span>Key</span>
<span>Aliases</span>
<span>Read</span> <span>Read</span>
<span>Write</span> <span>Write</span>
<span>Owner</span> <span>Owner</span>
@ -61,6 +62,7 @@ const PermissionsTab = () => {
<Table.Row> <Table.Row>
<span>{idx + 1}</span> <span>{idx + 1}</span>
<span>{key.name || key.accessKeyId?.substring(0, 8)}</span> <span>{key.name || key.accessKeyId?.substring(0, 8)}</span>
<span>{key.bucketLocalAliases?.join(", ") || "-"}</span>
<span> <span>
<Checkbox <Checkbox
checked={key.permissions?.read} checked={key.permissions?.read}

View File

@ -10,20 +10,27 @@ const BucketsPage = () => {
const [search, setSearch] = useState(""); const [search, setSearch] = useState("");
const items = useMemo(() => { const items = useMemo(() => {
let buckets = data || []; let buckets =
data?.map((bucket) => {
return {
...bucket,
aliases: [
...(bucket.globalAliases || []),
...(bucket.localAliases?.map((l) => l.alias) || []),
],
};
}) || [];
if (search?.length > 0) { if (search?.length > 0) {
const q = search.toLowerCase(); const q = search.toLowerCase();
buckets = buckets.filter( buckets = buckets.filter(
(bucket) => (bucket) =>
bucket.id.includes(q) || bucket.id.includes(q) ||
bucket.globalAliases.find((alias) => alias.includes(q)) bucket.aliases.find((alias) => alias.includes(q))
); );
} }
buckets = buckets.sort((a, b) => buckets = buckets.sort((a, b) => a.aliases[0].localeCompare(b.aliases[0]));
a.globalAliases[0].localeCompare(b.globalAliases[0])
);
return buckets; return buckets;
}, [data, search]); }, [data, search]);

View File

@ -5,6 +5,7 @@ export type GetBucketRes = Bucket[];
export type Bucket = { export type Bucket = {
id: string; id: string;
globalAliases: string[]; globalAliases: string[];
localAliases: LocalAlias[];
websiteAccess: boolean; websiteAccess: boolean;
websiteConfig?: WebsiteConfig | null; websiteConfig?: WebsiteConfig | null;
keys: Key[]; keys: Key[];
@ -17,11 +18,16 @@ export type Bucket = {
quotas: Quotas; quotas: Quotas;
}; };
export type LocalAlias = {
accessKeyId: string;
alias: string;
};
export type Key = { export type Key = {
accessKeyId: string; accessKeyId: string;
name: string; name: string;
permissions: Permissions; permissions: Permissions;
bucketLocalAliases: any[]; bucketLocalAliases: string[];
}; };
export type Permissions = { export type Permissions = {