-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathmulti-vendor-schema.sql
More file actions
365 lines (317 loc) · 11.6 KB
/
multi-vendor-schema.sql
File metadata and controls
365 lines (317 loc) · 11.6 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
-- Multi-Vendor Schema for SolMart
-- Complete implementation in a single file
-- ============================================================================
-- 1. EXTEND EXISTING TABLES FOR MULTI-VENDOR FUNCTIONALITY
-- ============================================================================
-- Extend users table for vendor functionality
ALTER TABLE public.users
ADD COLUMN IF NOT EXISTS business_name TEXT,
ADD COLUMN IF NOT EXISTS business_description TEXT,
ADD COLUMN IF NOT EXISTS phone TEXT,
ADD COLUMN IF NOT EXISTS website TEXT,
ADD COLUMN IF NOT EXISTS logo TEXT,
ADD COLUMN IF NOT EXISTS banner TEXT,
ADD COLUMN IF NOT EXISTS business_address JSONB,
ADD COLUMN IF NOT EXISTS vendor_status TEXT DEFAULT 'pending',
ADD COLUMN IF NOT EXISTS updated_at TIMESTAMP WITHOUT TIME ZONE DEFAULT now();
-- Add vendor status constraint
ALTER TABLE public.users
ADD CONSTRAINT users_vendor_status_check
CHECK (vendor_status IN ('pending', 'approved', 'rejected', 'suspended'));
-- Extend products table for multi-vendor
ALTER TABLE public.products
ADD COLUMN IF NOT EXISTS is_active BOOLEAN DEFAULT true,
ADD COLUMN IF NOT EXISTS updated_at TIMESTAMP WITHOUT TIME ZONE DEFAULT now(),
ADD COLUMN IF NOT EXISTS vendor_name TEXT;
-- Add product constraints
ALTER TABLE public.products
ADD CONSTRAINT products_is_active_check
CHECK (is_active IN (true, false));
-- Extend orders table for multi-vendor orders
ALTER TABLE public.orders
ADD COLUMN IF NOT EXISTS user_wallet TEXT,
ADD COLUMN IF NOT EXISTS user_email TEXT,
ADD COLUMN IF NOT EXISTS items JSONB,
ADD COLUMN IF NOT EXISTS total_amount NUMERIC,
ADD COLUMN IF NOT EXISTS transaction_signature TEXT,
ADD COLUMN IF NOT EXISTS tracking_number TEXT,
ADD COLUMN IF NOT EXISTS estimated_delivery TEXT,
ADD COLUMN IF NOT EXISTS shipping_address JSONB,
ADD COLUMN IF NOT EXISTS notes TEXT,
ADD COLUMN IF NOT EXISTS updated_at TIMESTAMP WITHOUT TIME ZONE DEFAULT now(),
ADD COLUMN IF NOT EXISTS vendor_orders JSONB DEFAULT '[]';
-- Add order status constraint
ALTER TABLE public.orders
ADD CONSTRAINT orders_status_check
CHECK (status IN ('pending', 'processing', 'shipped', 'delivered', 'cancelled'));
-- ============================================================================
-- 2. CREATE INDEXES FOR PERFORMANCE
-- ============================================================================
-- Products indexes
CREATE INDEX IF NOT EXISTS idx_products_seller_id ON public.products(seller_id);
CREATE INDEX IF NOT EXISTS idx_products_category ON public.products(category);
CREATE INDEX IF NOT EXISTS idx_products_is_active ON public.products(is_active);
CREATE INDEX IF NOT EXISTS idx_products_created_at ON public.products(created_at);
-- Orders indexes
CREATE INDEX IF NOT EXISTS idx_orders_user_wallet ON public.orders(user_wallet);
CREATE INDEX IF NOT EXISTS idx_orders_status ON public.orders(status);
CREATE INDEX IF NOT EXISTS idx_orders_created_at ON public.orders(created_at);
-- Users indexes
CREATE INDEX IF NOT EXISTS idx_users_wallet_address ON public.users(wallet_address);
CREATE INDEX IF NOT EXISTS idx_users_vendor_status ON public.users(vendor_status);
CREATE INDEX IF NOT EXISTS idx_users_role ON public.users(role);
-- ============================================================================
-- 3. ENABLE ROW LEVEL SECURITY
-- ============================================================================
ALTER TABLE public.products ENABLE ROW LEVEL SECURITY;
ALTER TABLE public.orders ENABLE ROW LEVEL SECURITY;
ALTER TABLE public.users ENABLE ROW LEVEL SECURITY;
-- ============================================================================
-- 4. CREATE RLS POLICIES
-- ============================================================================
-- Products policies
CREATE POLICY "Anyone can view active products from approved vendors" ON public.products
FOR SELECT USING (
is_active = true AND
seller_id IN (
SELECT id FROM public.users
WHERE vendor_status = 'approved' OR role = 'admin'
)
);
CREATE POLICY "Vendors can manage their own products" ON public.products
FOR ALL USING (
seller_id IN (
SELECT id FROM public.users
WHERE wallet_address = auth.jwt() ->> 'wallet_address'
)
);
CREATE POLICY "Admins can manage all products" ON public.products
FOR ALL USING (
EXISTS (
SELECT 1 FROM public.users
WHERE wallet_address = auth.jwt() ->> 'wallet_address'
AND role = 'admin'
)
);
-- Orders policies
CREATE POLICY "Users can view their own orders" ON public.orders
FOR SELECT USING (user_wallet = auth.jwt() ->> 'wallet_address');
CREATE POLICY "Users can create their own orders" ON public.orders
FOR INSERT WITH CHECK (user_wallet = auth.jwt() ->> 'wallet_address');
CREATE POLICY "Admins can view all orders" ON public.orders
FOR ALL USING (
EXISTS (
SELECT 1 FROM public.users
WHERE wallet_address = auth.jwt() ->> 'wallet_address'
AND role = 'admin'
)
);
-- Users policies
CREATE POLICY "Users can view their own profile" ON public.users
FOR SELECT USING (wallet_address = auth.jwt() ->> 'wallet_address');
CREATE POLICY "Users can update their own profile" ON public.users
FOR UPDATE USING (wallet_address = auth.jwt() ->> 'wallet_address');
CREATE POLICY "Anyone can create a user profile" ON public.users
FOR INSERT WITH CHECK (true);
-- ============================================================================
-- 5. CREATE HELPER FUNCTIONS
-- ============================================================================
-- Update timestamp function
CREATE OR REPLACE FUNCTION update_updated_at_column()
RETURNS TRIGGER AS $$
BEGIN
NEW.updated_at = NOW();
RETURN NEW;
END;
$$ language 'plpgsql';
-- Vendor status validation function
CREATE OR REPLACE FUNCTION validate_vendor_status()
RETURNS TRIGGER AS $$
BEGIN
IF NEW.seller_id IS NOT NULL THEN
IF NOT EXISTS (
SELECT 1 FROM public.users
WHERE id = NEW.seller_id
AND (vendor_status = 'approved' OR role = 'admin')
) THEN
RAISE EXCEPTION 'Only approved vendors can create/update products';
END IF;
END IF;
RETURN NEW;
END;
$$ language 'plpgsql';
-- Get vendor statistics function
CREATE OR REPLACE FUNCTION get_vendor_stats(vendor_id UUID)
RETURNS TABLE(
total_products BIGINT,
active_products BIGINT,
total_revenue NUMERIC,
total_orders BIGINT
) AS $$
BEGIN
RETURN QUERY
SELECT
COUNT(p.id)::BIGINT as total_products,
COUNT(CASE WHEN p.is_active = true THEN 1 END)::BIGINT as active_products,
COALESCE(SUM(CASE WHEN o.status = 'delivered' THEN o.total_amount ELSE 0 END), 0) as total_revenue,
COUNT(DISTINCT o.id)::BIGINT as total_orders
FROM public.users u
LEFT JOIN public.products p ON u.id = p.seller_id
LEFT JOIN public.orders o ON p.id = o.product_id
WHERE u.id = vendor_id;
END;
$$ language 'plpgsql';
-- Check if wallet is vendor function
CREATE OR REPLACE FUNCTION is_vendor(wallet_address TEXT)
RETURNS BOOLEAN AS $$
BEGIN
RETURN EXISTS (
SELECT 1 FROM public.users
WHERE wallet_address = $1
AND vendor_status = 'approved'
);
END;
$$ language 'plpgsql';
-- Get vendor by wallet function
CREATE OR REPLACE FUNCTION get_vendor_by_wallet(wallet_address TEXT)
RETURNS TABLE(
id UUID,
business_name TEXT,
vendor_status TEXT,
role TEXT
) AS $$
BEGIN
RETURN QUERY
SELECT
u.id,
u.business_name,
u.vendor_status,
u.role
FROM public.users u
WHERE u.wallet_address = $1;
END;
$$ language 'plpgsql';
-- Get approved vendors function
CREATE OR REPLACE FUNCTION get_approved_vendors()
RETURNS TABLE(
id UUID,
business_name TEXT,
vendor_status TEXT,
created_at TIMESTAMP
) AS $$
BEGIN
RETURN QUERY
SELECT
u.id,
u.business_name,
u.vendor_status,
u.created_at
FROM public.users u
WHERE u.vendor_status = 'approved'
ORDER BY u.created_at DESC;
END;
$$ language 'plpgsql';
-- ============================================================================
-- 6. CREATE TRIGGERS
-- ============================================================================
-- Update timestamp triggers
CREATE TRIGGER update_products_updated_at
BEFORE UPDATE ON public.products
FOR EACH ROW
EXECUTE FUNCTION update_updated_at_column();
CREATE TRIGGER update_orders_updated_at
BEFORE UPDATE ON public.orders
FOR EACH ROW
EXECUTE FUNCTION update_updated_at_column();
CREATE TRIGGER update_users_updated_at
BEFORE UPDATE ON public.users
FOR EACH ROW
EXECUTE FUNCTION update_updated_at_column();
-- Vendor status validation trigger
CREATE TRIGGER validate_vendor_status_trigger
BEFORE INSERT OR UPDATE ON public.products
FOR EACH ROW
EXECUTE FUNCTION validate_vendor_status();
-- Auto-populate vendor name trigger
CREATE OR REPLACE FUNCTION update_product_vendor_info()
RETURNS TRIGGER AS $$
BEGIN
IF NEW.seller_id IS NOT NULL THEN
SELECT business_name INTO NEW.vendor_name
FROM public.users
WHERE id = NEW.seller_id;
END IF;
RETURN NEW;
END;
$$ language 'plpgsql';
CREATE TRIGGER update_product_vendor_info_trigger
BEFORE INSERT OR UPDATE ON public.products
FOR EACH ROW
EXECUTE FUNCTION update_product_vendor_info();
-- ============================================================================
-- 7. CREATE VIEWS FOR EASY DATA ACCESS
-- ============================================================================
-- Products with vendor information
CREATE OR REPLACE VIEW products_with_vendors AS
SELECT
p.*,
u.business_name as vendor_name,
u.wallet_address as vendor_wallet,
u.vendor_status,
u.role
FROM public.products p
LEFT JOIN public.users u ON p.seller_id = u.id
WHERE p.is_active = true;
-- Vendor statistics view
CREATE OR REPLACE VIEW vendor_stats AS
SELECT
u.id as vendor_id,
u.business_name,
u.vendor_status,
COUNT(p.id) as total_products,
COUNT(CASE WHEN p.is_active = true THEN 1 END) as active_products,
COALESCE(SUM(CASE WHEN o.status = 'delivered' THEN o.total_amount ELSE 0 END), 0) as total_revenue,
COUNT(DISTINCT o.id) as total_orders
FROM public.users u
LEFT JOIN public.products p ON u.id = p.seller_id
LEFT JOIN public.orders o ON p.id = o.product_id
WHERE u.vendor_status = 'approved'
GROUP BY u.id, u.business_name, u.vendor_status;
-- ============================================================================
-- 8. DATA MIGRATION
-- ============================================================================
-- Migrate existing order data
UPDATE public.orders
SET user_wallet = buyer_wallet
WHERE user_wallet IS NULL AND buyer_wallet IS NOT NULL;
UPDATE public.orders
SET total_amount = total_price_usdc
WHERE total_amount IS NULL AND total_price_usdc IS NOT NULL;
-- Migrate existing product data
UPDATE public.products p
SET vendor_name = u.business_name
FROM public.users u
WHERE p.seller_id = u.id AND p.vendor_name IS NULL;
-- Set default vendor status for existing users
UPDATE public.users
SET vendor_status = 'approved'
WHERE vendor_status IS NULL AND role = 'admin';
UPDATE public.users
SET vendor_status = 'pending'
WHERE vendor_status IS NULL AND role = 'buyer';
-- ============================================================================
-- 9. SUCCESS MESSAGE
-- ============================================================================
DO $$
BEGIN
RAISE NOTICE 'Multi-vendor schema implementation completed successfully!';
RAISE NOTICE 'Features added:';
RAISE NOTICE '- Vendor management in users table';
RAISE NOTICE '- Multi-vendor product support';
RAISE NOTICE '- Enhanced order tracking';
RAISE NOTICE '- Row Level Security policies';
RAISE NOTICE '- Performance indexes';
RAISE NOTICE '- Helper functions and triggers';
RAISE NOTICE '- Data views for easy access';
RAISE NOTICE '- Existing data migration completed';
END $$;