|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CREATE EXTENSION IF NOT EXISTS "uuid-ossp"; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
IF EXISTS ( |
|
|
SELECT 1 FROM information_schema.table_constraints |
|
|
WHERE constraint_name = 'fk_version' |
|
|
AND table_name = 'download_logs' |
|
|
) THEN |
|
|
ALTER TABLE download_logs DROP CONSTRAINT fk_version; |
|
|
END IF; |
|
|
END $$; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CREATE TABLE IF NOT EXISTS versions ( |
|
|
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(), |
|
|
version VARCHAR(20) NOT NULL UNIQUE, |
|
|
title VARCHAR(255), |
|
|
release_date TIMESTAMP WITH TIME ZONE DEFAULT NOW(), |
|
|
is_active BOOLEAN DEFAULT true, |
|
|
download_url TEXT NOT NULL, |
|
|
file_size BIGINT, |
|
|
file_type VARCHAR(10) CHECK (file_type IN ('zip', 'exe', 'msi')) DEFAULT 'zip', |
|
|
file_name TEXT NOT NULL, |
|
|
changelog TEXT, |
|
|
min_revit_version VARCHAR(10), |
|
|
released_by VARCHAR(255), |
|
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(), |
|
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW() |
|
|
); |
|
|
|
|
|
|
|
|
CREATE TABLE IF NOT EXISTS download_logs ( |
|
|
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(), |
|
|
version VARCHAR(20), |
|
|
license_key VARCHAR(255), |
|
|
action VARCHAR(20) CHECK (action IN ('download', 'update_check', 'install')), |
|
|
user_agent TEXT, |
|
|
ip_address INET, |
|
|
downloaded_at TIMESTAMP WITH TIME ZONE DEFAULT NOW() |
|
|
); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CREATE INDEX IF NOT EXISTS idx_versions_version ON versions(version); |
|
|
CREATE INDEX IF NOT EXISTS idx_versions_release_date ON versions(release_date DESC); |
|
|
CREATE INDEX IF NOT EXISTS idx_versions_is_active ON versions(is_active); |
|
|
|
|
|
|
|
|
CREATE INDEX IF NOT EXISTS idx_download_logs_version ON download_logs(version); |
|
|
CREATE INDEX IF NOT EXISTS idx_download_logs_downloaded_at ON download_logs(downloaded_at DESC); |
|
|
CREATE INDEX IF NOT EXISTS idx_download_logs_action ON download_logs(action); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
ALTER TABLE versions ENABLE ROW LEVEL SECURITY; |
|
|
EXCEPTION |
|
|
WHEN OTHERS THEN NULL; |
|
|
END $$; |
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
ALTER TABLE download_logs ENABLE ROW LEVEL SECURITY; |
|
|
EXCEPTION |
|
|
WHEN OTHERS THEN NULL; |
|
|
END $$; |
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
|
|
|
DROP POLICY IF EXISTS "Public read versions" ON versions; |
|
|
CREATE POLICY "Public read versions" ON versions |
|
|
FOR SELECT USING (true); |
|
|
|
|
|
DROP POLICY IF EXISTS "Service role insert versions" ON versions; |
|
|
CREATE POLICY "Service role insert versions" ON versions |
|
|
FOR INSERT WITH CHECK (auth.role() = 'service_role'); |
|
|
|
|
|
DROP POLICY IF EXISTS "Service role update versions" ON versions; |
|
|
CREATE POLICY "Service role update versions" ON versions |
|
|
FOR UPDATE USING (auth.role() = 'service_role'); |
|
|
|
|
|
|
|
|
DROP POLICY IF EXISTS "Service role all download_logs" ON download_logs; |
|
|
CREATE POLICY "Service role all download_logs" ON download_logs |
|
|
FOR ALL USING (auth.role() = 'service_role'); |
|
|
END $$; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
INSERT INTO storage.buckets (id, name, public) |
|
|
VALUES ('plugin-releases', 'plugin-releases', true); |
|
|
EXCEPTION |
|
|
WHEN unique_violation THEN NULL; |
|
|
END $$; |
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
|
|
|
DROP POLICY IF EXISTS "Public read plugin releases" ON storage.objects; |
|
|
CREATE POLICY "Public read plugin releases" ON storage.objects |
|
|
FOR SELECT USING (bucket_id = 'plugin-releases'); |
|
|
|
|
|
|
|
|
DROP POLICY IF EXISTS "Service role manage plugin releases" ON storage.objects; |
|
|
CREATE POLICY "Service role manage plugin releases" ON storage.objects |
|
|
FOR ALL USING (auth.role() = 'service_role' AND bucket_id = 'plugin-releases'); |
|
|
|
|
|
|
|
|
DROP POLICY IF EXISTS "Authenticated users upload plugin releases" ON storage.objects; |
|
|
CREATE POLICY "Authenticated users upload plugin releases" ON storage.objects |
|
|
FOR INSERT WITH CHECK ( |
|
|
bucket_id = 'plugin-releases' AND |
|
|
auth.role() = 'authenticated' |
|
|
); |
|
|
END $$; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CREATE OR REPLACE FUNCTION update_updated_at_column() |
|
|
RETURNS TRIGGER AS $$ |
|
|
BEGIN |
|
|
NEW.updated_at = NOW(); |
|
|
RETURN NEW; |
|
|
END; |
|
|
$$ language 'plpgsql'; |
|
|
|
|
|
|
|
|
DO $$ |
|
|
BEGIN |
|
|
DROP TRIGGER IF EXISTS update_versions_updated_at ON versions; |
|
|
CREATE TRIGGER update_versions_updated_at |
|
|
BEFORE UPDATE ON versions |
|
|
FOR EACH ROW EXECUTE FUNCTION update_updated_at_column(); |
|
|
END $$; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DROP VIEW IF EXISTS latest_version; |
|
|
CREATE VIEW latest_version AS |
|
|
SELECT * |
|
|
FROM versions |
|
|
WHERE is_active = true |
|
|
ORDER BY release_date DESC |
|
|
LIMIT 1; |
|
|
|
|
|
DROP VIEW IF EXISTS version_stats; |
|
|
CREATE VIEW version_stats AS |
|
|
SELECT |
|
|
v.version, |
|
|
v.release_date, |
|
|
COUNT(dl.id) as download_count, |
|
|
COUNT(CASE WHEN dl.action = 'download' THEN 1 END) as actual_downloads, |
|
|
COUNT(CASE WHEN dl.action = 'update_check' THEN 1 END) as update_checks |
|
|
FROM versions v |
|
|
LEFT JOIN download_logs dl ON v.version = dl.version |
|
|
WHERE v.is_active = true |
|
|
GROUP BY v.id, v.version, v.release_date |
|
|
ORDER BY v.release_date DESC; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
COMMENT ON TABLE versions IS '插件版本資訊表'; |
|
|
COMMENT ON COLUMN versions.version IS '版本號,格式:x.y.z'; |
|
|
COMMENT ON COLUMN versions.title IS '版本標題,簡短描述版本特色'; |
|
|
COMMENT ON COLUMN versions.download_url IS '下載連結URL'; |
|
|
COMMENT ON COLUMN versions.file_size IS '檔案大小(位元組)'; |
|
|
COMMENT ON COLUMN versions.file_type IS '檔案類型:zip, exe, msi'; |
|
|
COMMENT ON COLUMN versions.file_name IS '檔案名稱,包含副檔名'; |
|
|
COMMENT ON COLUMN versions.changelog IS '更新日誌(Markdown格式)'; |
|
|
COMMENT ON COLUMN versions.min_revit_version IS '最小支援的Revit版本'; |
|
|
|
|
|
COMMENT ON TABLE download_logs IS '下載記錄表'; |
|
|
COMMENT ON COLUMN download_logs.action IS '操作類型:download, update_check, install'; |