mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-19 09:58:38 +00:00
refactor(shared): dedupe frontmatter parsing
This commit is contained in:
@@ -1,5 +1,4 @@
|
|||||||
import type { Skill } from "@mariozechner/pi-coding-agent";
|
import type { Skill } from "@mariozechner/pi-coding-agent";
|
||||||
import JSON5 from "json5";
|
|
||||||
import type {
|
import type {
|
||||||
OpenClawSkillMetadata,
|
OpenClawSkillMetadata,
|
||||||
ParsedSkillFrontmatter,
|
ParsedSkillFrontmatter,
|
||||||
@@ -7,30 +6,18 @@ import type {
|
|||||||
SkillInstallSpec,
|
SkillInstallSpec,
|
||||||
SkillInvocationPolicy,
|
SkillInvocationPolicy,
|
||||||
} from "./types.js";
|
} from "./types.js";
|
||||||
import { LEGACY_MANIFEST_KEYS, MANIFEST_KEY } from "../../compat/legacy-names.js";
|
|
||||||
import { parseFrontmatterBlock } from "../../markdown/frontmatter.js";
|
import { parseFrontmatterBlock } from "../../markdown/frontmatter.js";
|
||||||
import { parseBooleanValue } from "../../utils/boolean.js";
|
import {
|
||||||
|
getFrontmatterString,
|
||||||
|
normalizeStringList,
|
||||||
|
parseFrontmatterBool,
|
||||||
|
resolveOpenClawManifestBlock,
|
||||||
|
} from "../../shared/frontmatter.js";
|
||||||
|
|
||||||
export function parseFrontmatter(content: string): ParsedSkillFrontmatter {
|
export function parseFrontmatter(content: string): ParsedSkillFrontmatter {
|
||||||
return parseFrontmatterBlock(content);
|
return parseFrontmatterBlock(content);
|
||||||
}
|
}
|
||||||
|
|
||||||
function normalizeStringList(input: unknown): string[] {
|
|
||||||
if (!input) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
if (Array.isArray(input)) {
|
|
||||||
return input.map((value) => String(value).trim()).filter(Boolean);
|
|
||||||
}
|
|
||||||
if (typeof input === "string") {
|
|
||||||
return input
|
|
||||||
.split(",")
|
|
||||||
.map((value) => value.trim())
|
|
||||||
.filter(Boolean);
|
|
||||||
}
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseInstallSpec(input: unknown): SkillInstallSpec | undefined {
|
function parseInstallSpec(input: unknown): SkillInstallSpec | undefined {
|
||||||
if (!input || typeof input !== "object") {
|
if (!input || typeof input !== "object") {
|
||||||
return undefined;
|
return undefined;
|
||||||
@@ -89,79 +76,48 @@ function parseInstallSpec(input: unknown): SkillInstallSpec | undefined {
|
|||||||
return spec;
|
return spec;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getFrontmatterValue(frontmatter: ParsedSkillFrontmatter, key: string): string | undefined {
|
|
||||||
const raw = frontmatter[key];
|
|
||||||
return typeof raw === "string" ? raw : undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseFrontmatterBool(value: string | undefined, fallback: boolean): boolean {
|
|
||||||
const parsed = parseBooleanValue(value);
|
|
||||||
return parsed === undefined ? fallback : parsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function resolveOpenClawMetadata(
|
export function resolveOpenClawMetadata(
|
||||||
frontmatter: ParsedSkillFrontmatter,
|
frontmatter: ParsedSkillFrontmatter,
|
||||||
): OpenClawSkillMetadata | undefined {
|
): OpenClawSkillMetadata | undefined {
|
||||||
const raw = getFrontmatterValue(frontmatter, "metadata");
|
const metadataObj = resolveOpenClawManifestBlock({ frontmatter });
|
||||||
if (!raw) {
|
if (!metadataObj) {
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const parsed = JSON5.parse(raw);
|
|
||||||
if (!parsed || typeof parsed !== "object") {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
const metadataRawCandidates = [MANIFEST_KEY, ...LEGACY_MANIFEST_KEYS];
|
|
||||||
let metadataRaw: unknown;
|
|
||||||
for (const key of metadataRawCandidates) {
|
|
||||||
const candidate = parsed[key];
|
|
||||||
if (candidate && typeof candidate === "object") {
|
|
||||||
metadataRaw = candidate;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!metadataRaw || typeof metadataRaw !== "object") {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
const metadataObj = metadataRaw as Record<string, unknown>;
|
|
||||||
const requiresRaw =
|
|
||||||
typeof metadataObj.requires === "object" && metadataObj.requires !== null
|
|
||||||
? (metadataObj.requires as Record<string, unknown>)
|
|
||||||
: undefined;
|
|
||||||
const installRaw = Array.isArray(metadataObj.install) ? (metadataObj.install as unknown[]) : [];
|
|
||||||
const install = installRaw
|
|
||||||
.map((entry) => parseInstallSpec(entry))
|
|
||||||
.filter((entry): entry is SkillInstallSpec => Boolean(entry));
|
|
||||||
const osRaw = normalizeStringList(metadataObj.os);
|
|
||||||
return {
|
|
||||||
always: typeof metadataObj.always === "boolean" ? metadataObj.always : undefined,
|
|
||||||
emoji: typeof metadataObj.emoji === "string" ? metadataObj.emoji : undefined,
|
|
||||||
homepage: typeof metadataObj.homepage === "string" ? metadataObj.homepage : undefined,
|
|
||||||
skillKey: typeof metadataObj.skillKey === "string" ? metadataObj.skillKey : undefined,
|
|
||||||
primaryEnv: typeof metadataObj.primaryEnv === "string" ? metadataObj.primaryEnv : undefined,
|
|
||||||
os: osRaw.length > 0 ? osRaw : undefined,
|
|
||||||
requires: requiresRaw
|
|
||||||
? {
|
|
||||||
bins: normalizeStringList(requiresRaw.bins),
|
|
||||||
anyBins: normalizeStringList(requiresRaw.anyBins),
|
|
||||||
env: normalizeStringList(requiresRaw.env),
|
|
||||||
config: normalizeStringList(requiresRaw.config),
|
|
||||||
}
|
|
||||||
: undefined,
|
|
||||||
install: install.length > 0 ? install : undefined,
|
|
||||||
};
|
|
||||||
} catch {
|
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
const requiresRaw =
|
||||||
|
typeof metadataObj.requires === "object" && metadataObj.requires !== null
|
||||||
|
? (metadataObj.requires as Record<string, unknown>)
|
||||||
|
: undefined;
|
||||||
|
const installRaw = Array.isArray(metadataObj.install) ? (metadataObj.install as unknown[]) : [];
|
||||||
|
const install = installRaw
|
||||||
|
.map((entry) => parseInstallSpec(entry))
|
||||||
|
.filter((entry): entry is SkillInstallSpec => Boolean(entry));
|
||||||
|
const osRaw = normalizeStringList(metadataObj.os);
|
||||||
|
return {
|
||||||
|
always: typeof metadataObj.always === "boolean" ? metadataObj.always : undefined,
|
||||||
|
emoji: typeof metadataObj.emoji === "string" ? metadataObj.emoji : undefined,
|
||||||
|
homepage: typeof metadataObj.homepage === "string" ? metadataObj.homepage : undefined,
|
||||||
|
skillKey: typeof metadataObj.skillKey === "string" ? metadataObj.skillKey : undefined,
|
||||||
|
primaryEnv: typeof metadataObj.primaryEnv === "string" ? metadataObj.primaryEnv : undefined,
|
||||||
|
os: osRaw.length > 0 ? osRaw : undefined,
|
||||||
|
requires: requiresRaw
|
||||||
|
? {
|
||||||
|
bins: normalizeStringList(requiresRaw.bins),
|
||||||
|
anyBins: normalizeStringList(requiresRaw.anyBins),
|
||||||
|
env: normalizeStringList(requiresRaw.env),
|
||||||
|
config: normalizeStringList(requiresRaw.config),
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
|
install: install.length > 0 ? install : undefined,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function resolveSkillInvocationPolicy(
|
export function resolveSkillInvocationPolicy(
|
||||||
frontmatter: ParsedSkillFrontmatter,
|
frontmatter: ParsedSkillFrontmatter,
|
||||||
): SkillInvocationPolicy {
|
): SkillInvocationPolicy {
|
||||||
return {
|
return {
|
||||||
userInvocable: parseFrontmatterBool(getFrontmatterValue(frontmatter, "user-invocable"), true),
|
userInvocable: parseFrontmatterBool(getFrontmatterString(frontmatter, "user-invocable"), true),
|
||||||
disableModelInvocation: parseFrontmatterBool(
|
disableModelInvocation: parseFrontmatterBool(
|
||||||
getFrontmatterValue(frontmatter, "disable-model-invocation"),
|
getFrontmatterString(frontmatter, "disable-model-invocation"),
|
||||||
false,
|
false,
|
||||||
),
|
),
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import JSON5 from "json5";
|
|
||||||
import type {
|
import type {
|
||||||
OpenClawHookMetadata,
|
OpenClawHookMetadata,
|
||||||
HookEntry,
|
HookEntry,
|
||||||
@@ -6,30 +5,18 @@ import type {
|
|||||||
HookInvocationPolicy,
|
HookInvocationPolicy,
|
||||||
ParsedHookFrontmatter,
|
ParsedHookFrontmatter,
|
||||||
} from "./types.js";
|
} from "./types.js";
|
||||||
import { LEGACY_MANIFEST_KEYS, MANIFEST_KEY } from "../compat/legacy-names.js";
|
|
||||||
import { parseFrontmatterBlock } from "../markdown/frontmatter.js";
|
import { parseFrontmatterBlock } from "../markdown/frontmatter.js";
|
||||||
import { parseBooleanValue } from "../utils/boolean.js";
|
import {
|
||||||
|
getFrontmatterString,
|
||||||
|
normalizeStringList,
|
||||||
|
parseFrontmatterBool,
|
||||||
|
resolveOpenClawManifestBlock,
|
||||||
|
} from "../shared/frontmatter.js";
|
||||||
|
|
||||||
export function parseFrontmatter(content: string): ParsedHookFrontmatter {
|
export function parseFrontmatter(content: string): ParsedHookFrontmatter {
|
||||||
return parseFrontmatterBlock(content);
|
return parseFrontmatterBlock(content);
|
||||||
}
|
}
|
||||||
|
|
||||||
function normalizeStringList(input: unknown): string[] {
|
|
||||||
if (!input) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
if (Array.isArray(input)) {
|
|
||||||
return input.map((value) => String(value).trim()).filter(Boolean);
|
|
||||||
}
|
|
||||||
if (typeof input === "string") {
|
|
||||||
return input
|
|
||||||
.split(",")
|
|
||||||
.map((value) => value.trim())
|
|
||||||
.filter(Boolean);
|
|
||||||
}
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseInstallSpec(input: unknown): HookInstallSpec | undefined {
|
function parseInstallSpec(input: unknown): HookInstallSpec | undefined {
|
||||||
if (!input || typeof input !== "object") {
|
if (!input || typeof input !== "object") {
|
||||||
return undefined;
|
return undefined;
|
||||||
@@ -66,79 +53,48 @@ function parseInstallSpec(input: unknown): HookInstallSpec | undefined {
|
|||||||
return spec;
|
return spec;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getFrontmatterValue(frontmatter: ParsedHookFrontmatter, key: string): string | undefined {
|
|
||||||
const raw = frontmatter[key];
|
|
||||||
return typeof raw === "string" ? raw : undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseFrontmatterBool(value: string | undefined, fallback: boolean): boolean {
|
|
||||||
const parsed = parseBooleanValue(value);
|
|
||||||
return parsed === undefined ? fallback : parsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function resolveOpenClawMetadata(
|
export function resolveOpenClawMetadata(
|
||||||
frontmatter: ParsedHookFrontmatter,
|
frontmatter: ParsedHookFrontmatter,
|
||||||
): OpenClawHookMetadata | undefined {
|
): OpenClawHookMetadata | undefined {
|
||||||
const raw = getFrontmatterValue(frontmatter, "metadata");
|
const metadataObj = resolveOpenClawManifestBlock({ frontmatter });
|
||||||
if (!raw) {
|
if (!metadataObj) {
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const parsed = JSON5.parse(raw);
|
|
||||||
if (!parsed || typeof parsed !== "object") {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
const metadataRawCandidates = [MANIFEST_KEY, ...LEGACY_MANIFEST_KEYS];
|
|
||||||
let metadataRaw: unknown;
|
|
||||||
for (const key of metadataRawCandidates) {
|
|
||||||
const candidate = parsed[key];
|
|
||||||
if (candidate && typeof candidate === "object") {
|
|
||||||
metadataRaw = candidate;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!metadataRaw || typeof metadataRaw !== "object") {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
const metadataObj = metadataRaw as Record<string, unknown>;
|
|
||||||
const requiresRaw =
|
|
||||||
typeof metadataObj.requires === "object" && metadataObj.requires !== null
|
|
||||||
? (metadataObj.requires as Record<string, unknown>)
|
|
||||||
: undefined;
|
|
||||||
const installRaw = Array.isArray(metadataObj.install) ? (metadataObj.install as unknown[]) : [];
|
|
||||||
const install = installRaw
|
|
||||||
.map((entry) => parseInstallSpec(entry))
|
|
||||||
.filter((entry): entry is HookInstallSpec => Boolean(entry));
|
|
||||||
const osRaw = normalizeStringList(metadataObj.os);
|
|
||||||
const eventsRaw = normalizeStringList(metadataObj.events);
|
|
||||||
return {
|
|
||||||
always: typeof metadataObj.always === "boolean" ? metadataObj.always : undefined,
|
|
||||||
emoji: typeof metadataObj.emoji === "string" ? metadataObj.emoji : undefined,
|
|
||||||
homepage: typeof metadataObj.homepage === "string" ? metadataObj.homepage : undefined,
|
|
||||||
hookKey: typeof metadataObj.hookKey === "string" ? metadataObj.hookKey : undefined,
|
|
||||||
export: typeof metadataObj.export === "string" ? metadataObj.export : undefined,
|
|
||||||
os: osRaw.length > 0 ? osRaw : undefined,
|
|
||||||
events: eventsRaw.length > 0 ? eventsRaw : [],
|
|
||||||
requires: requiresRaw
|
|
||||||
? {
|
|
||||||
bins: normalizeStringList(requiresRaw.bins),
|
|
||||||
anyBins: normalizeStringList(requiresRaw.anyBins),
|
|
||||||
env: normalizeStringList(requiresRaw.env),
|
|
||||||
config: normalizeStringList(requiresRaw.config),
|
|
||||||
}
|
|
||||||
: undefined,
|
|
||||||
install: install.length > 0 ? install : undefined,
|
|
||||||
};
|
|
||||||
} catch {
|
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
const requiresRaw =
|
||||||
|
typeof metadataObj.requires === "object" && metadataObj.requires !== null
|
||||||
|
? (metadataObj.requires as Record<string, unknown>)
|
||||||
|
: undefined;
|
||||||
|
const installRaw = Array.isArray(metadataObj.install) ? (metadataObj.install as unknown[]) : [];
|
||||||
|
const install = installRaw
|
||||||
|
.map((entry) => parseInstallSpec(entry))
|
||||||
|
.filter((entry): entry is HookInstallSpec => Boolean(entry));
|
||||||
|
const osRaw = normalizeStringList(metadataObj.os);
|
||||||
|
const eventsRaw = normalizeStringList(metadataObj.events);
|
||||||
|
return {
|
||||||
|
always: typeof metadataObj.always === "boolean" ? metadataObj.always : undefined,
|
||||||
|
emoji: typeof metadataObj.emoji === "string" ? metadataObj.emoji : undefined,
|
||||||
|
homepage: typeof metadataObj.homepage === "string" ? metadataObj.homepage : undefined,
|
||||||
|
hookKey: typeof metadataObj.hookKey === "string" ? metadataObj.hookKey : undefined,
|
||||||
|
export: typeof metadataObj.export === "string" ? metadataObj.export : undefined,
|
||||||
|
os: osRaw.length > 0 ? osRaw : undefined,
|
||||||
|
events: eventsRaw.length > 0 ? eventsRaw : [],
|
||||||
|
requires: requiresRaw
|
||||||
|
? {
|
||||||
|
bins: normalizeStringList(requiresRaw.bins),
|
||||||
|
anyBins: normalizeStringList(requiresRaw.anyBins),
|
||||||
|
env: normalizeStringList(requiresRaw.env),
|
||||||
|
config: normalizeStringList(requiresRaw.config),
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
|
install: install.length > 0 ? install : undefined,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function resolveHookInvocationPolicy(
|
export function resolveHookInvocationPolicy(
|
||||||
frontmatter: ParsedHookFrontmatter,
|
frontmatter: ParsedHookFrontmatter,
|
||||||
): HookInvocationPolicy {
|
): HookInvocationPolicy {
|
||||||
return {
|
return {
|
||||||
enabled: parseFrontmatterBool(getFrontmatterValue(frontmatter, "enabled"), true),
|
enabled: parseFrontmatterBool(getFrontmatterString(frontmatter, "enabled"), true),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
43
src/shared/frontmatter.test.ts
Normal file
43
src/shared/frontmatter.test.ts
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
import { describe, expect, test } from "vitest";
|
||||||
|
import {
|
||||||
|
getFrontmatterString,
|
||||||
|
normalizeStringList,
|
||||||
|
parseFrontmatterBool,
|
||||||
|
resolveOpenClawManifestBlock,
|
||||||
|
} from "./frontmatter.js";
|
||||||
|
|
||||||
|
describe("shared/frontmatter", () => {
|
||||||
|
test("normalizeStringList handles strings and arrays", () => {
|
||||||
|
expect(normalizeStringList("a, b,,c")).toEqual(["a", "b", "c"]);
|
||||||
|
expect(normalizeStringList([" a ", "", "b"])).toEqual(["a", "b"]);
|
||||||
|
expect(normalizeStringList(null)).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getFrontmatterString extracts strings only", () => {
|
||||||
|
expect(getFrontmatterString({ a: "b" }, "a")).toBe("b");
|
||||||
|
expect(getFrontmatterString({ a: 1 }, "a")).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("parseFrontmatterBool respects fallback", () => {
|
||||||
|
expect(parseFrontmatterBool("true", false)).toBe(true);
|
||||||
|
expect(parseFrontmatterBool("false", true)).toBe(false);
|
||||||
|
expect(parseFrontmatterBool(undefined, true)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("resolveOpenClawManifestBlock parses JSON5 metadata and picks openclaw block", () => {
|
||||||
|
const frontmatter = {
|
||||||
|
metadata: "{ openclaw: { foo: 1, bar: 'baz' } }",
|
||||||
|
};
|
||||||
|
expect(resolveOpenClawManifestBlock({ frontmatter })).toEqual({ foo: 1, bar: "baz" });
|
||||||
|
});
|
||||||
|
|
||||||
|
test("resolveOpenClawManifestBlock returns undefined for invalid input", () => {
|
||||||
|
expect(resolveOpenClawManifestBlock({ frontmatter: {} })).toBeUndefined();
|
||||||
|
expect(
|
||||||
|
resolveOpenClawManifestBlock({ frontmatter: { metadata: "not-json5" } }),
|
||||||
|
).toBeUndefined();
|
||||||
|
expect(
|
||||||
|
resolveOpenClawManifestBlock({ frontmatter: { metadata: "{ nope: { a: 1 } }" } }),
|
||||||
|
).toBeUndefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
60
src/shared/frontmatter.ts
Normal file
60
src/shared/frontmatter.ts
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import JSON5 from "json5";
|
||||||
|
import { LEGACY_MANIFEST_KEYS, MANIFEST_KEY } from "../compat/legacy-names.js";
|
||||||
|
import { parseBooleanValue } from "../utils/boolean.js";
|
||||||
|
|
||||||
|
export function normalizeStringList(input: unknown): string[] {
|
||||||
|
if (!input) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
if (Array.isArray(input)) {
|
||||||
|
return input.map((value) => String(value).trim()).filter(Boolean);
|
||||||
|
}
|
||||||
|
if (typeof input === "string") {
|
||||||
|
return input
|
||||||
|
.split(",")
|
||||||
|
.map((value) => value.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getFrontmatterString(
|
||||||
|
frontmatter: Record<string, unknown>,
|
||||||
|
key: string,
|
||||||
|
): string | undefined {
|
||||||
|
const raw = frontmatter[key];
|
||||||
|
return typeof raw === "string" ? raw : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseFrontmatterBool(value: string | undefined, fallback: boolean): boolean {
|
||||||
|
const parsed = parseBooleanValue(value);
|
||||||
|
return parsed === undefined ? fallback : parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function resolveOpenClawManifestBlock(params: {
|
||||||
|
frontmatter: Record<string, unknown>;
|
||||||
|
key?: string;
|
||||||
|
}): Record<string, unknown> | undefined {
|
||||||
|
const raw = getFrontmatterString(params.frontmatter, params.key ?? "metadata");
|
||||||
|
if (!raw) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsed = JSON5.parse(raw);
|
||||||
|
if (!parsed || typeof parsed !== "object") {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const manifestKeys = [MANIFEST_KEY, ...LEGACY_MANIFEST_KEYS];
|
||||||
|
for (const key of manifestKeys) {
|
||||||
|
const candidate = (parsed as Record<string, unknown>)[key];
|
||||||
|
if (candidate && typeof candidate === "object") {
|
||||||
|
return candidate as Record<string, unknown>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return undefined;
|
||||||
|
} catch {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user