Marketing Hub: RBAC, i18n (AR/EN), tasks overhaul, team/user merge, tutorial

Features:
- Full RBAC with 3 roles (superadmin/manager/contributor)
- Ownership tracking on posts, tasks, campaigns, projects
- Task system: assign to anyone, filter combobox, visibility scoping
- Team members merged into users table (single source of truth)
- Post thumbnails on kanban cards from attachments
- Publication link validation before publishing
- Interactive onboarding tutorial with Settings restart
- Full Arabic/English i18n with RTL layout support
- Language toggle in sidebar, IBM Plex Sans Arabic font
- Brand-based visibility filtering for non-superadmins
- Manager can only create contributors
- Profile completion flow for new users
- Cookie-based sessions (express-session + SQLite)
This commit is contained in:
fahed
2026-02-08 20:46:58 +03:00
commit 35d84b6bff
2240 changed files with 846749 additions and 0 deletions

182
server/node_modules/connect-sqlite3/test/mocha.css generated vendored Normal file
View File

@@ -0,0 +1,182 @@
body {
font: 20px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif;
padding: 60px 50px;
}
#mocha ul, #mocha li {
margin: 0;
padding: 0;
}
#mocha ul {
list-style: none;
}
#mocha h1, #mocha h2 {
margin: 0;
}
#mocha h1 {
margin-top: 15px;
font-size: 1em;
font-weight: 200;
}
#mocha h1 a {
text-decoration: none;
color: inherit;
}
#mocha h1 a:hover {
text-decoration: underline;
}
#mocha .suite .suite h1 {
margin-top: 0;
font-size: .8em;
}
#mocha h2 {
font-size: 12px;
font-weight: normal;
cursor: pointer;
}
#mocha .suite {
margin-left: 15px;
}
#mocha .test {
margin-left: 15px;
}
#mocha .test:hover h2::after {
position: relative;
top: 0;
right: -10px;
content: '(view source)';
font-size: 12px;
font-family: arial;
color: #888;
}
#mocha .test.pending:hover h2::after {
content: '(pending)';
font-family: arial;
}
#mocha .test.pass.medium .duration {
background: #C09853;
}
#mocha .test.pass.slow .duration {
background: #B94A48;
}
#mocha .test.pass::before {
content: '✓';
font-size: 12px;
display: block;
float: left;
margin-right: 5px;
color: #00d6b2;
}
#mocha .test.pass .duration {
font-size: 9px;
margin-left: 5px;
padding: 2px 5px;
color: white;
-webkit-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
-moz-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
-webkit-border-radius: 5px;
-moz-border-radius: 5px;
-ms-border-radius: 5px;
-o-border-radius: 5px;
border-radius: 5px;
}
#mocha .test.pass.fast .duration {
display: none;
}
#mocha .test.pending {
color: #0b97c4;
}
#mocha .test.pending::before {
content: '◦';
color: #0b97c4;
}
#mocha .test.fail {
color: #c00;
}
#mocha .test.fail pre {
color: black;
}
#mocha .test.fail::before {
content: '✖';
font-size: 12px;
display: block;
float: left;
margin-right: 5px;
color: #c00;
}
#mocha .test pre.error {
color: #c00;
}
#mocha .test pre {
display: inline-block;
font: 12px/1.5 monaco, monospace;
margin: 5px;
padding: 15px;
border: 1px solid #eee;
border-bottom-color: #ddd;
-webkit-border-radius: 3px;
-webkit-box-shadow: 0 1px 3px #eee;
}
#error {
color: #c00;
font-size: 1.5 em;
font-weight: 100;
letter-spacing: 1px;
}
#stats {
position: fixed;
top: 15px;
right: 10px;
font-size: 12px;
margin: 0;
color: #888;
}
#stats .progress {
float: right;
padding-top: 0;
}
#stats em {
color: black;
}
#stats li {
display: inline-block;
margin: 0 5px;
list-style: none;
padding-top: 11px;
}
code .comment { color: #ddd }
code .init { color: #2F6FAD }
code .string { color: #5890AD }
code .keyword { color: #8A6343 }
code .number { color: #2F6FAD }

4504
server/node_modules/connect-sqlite3/test/mocha.js generated vendored Normal file

File diff suppressed because it is too large Load Diff

124
server/node_modules/connect-sqlite3/test/test.js generated vendored Normal file
View File

@@ -0,0 +1,124 @@
var should = require('should'),
session = require('express-session'),
util = require('util'),
SQLiteStore = require('../lib/connect-sqlite3.js')(session);
describe('connect-sqlite3 basic test suite', function() {
before(function() {
this.memStore = new SQLiteStore({db: ':memory:', dir: 'dbs'});
});
after(function() {
//this.memStore.close();
});
it('it should save a new session record', function(done) {
this.memStore.set('1111222233334444', {cookie: {maxAge:2000}, name: 'sample name'}, function(err, rows) {
should.not.exist(err, 'set() returned an error');
rows.should.be.empty;
done();
});
});
it('it should retrieve an active session', function(done) {
this.memStore.get('1111222233334444', function(err, session) {
should.not.exist(err, 'get() returned an error');
should.exist(session);
(session).should.eql({cookie: {maxAge:2000}, name: 'sample name'});
done();
});
});
it('it should gracefully handle retrieving an unkonwn session', function(done) {
this.memStore.get('hope-and-change', function(err, rows) {
should.not.exist(err, 'get() unknown session returned an error');
should.equal(undefined, rows, 'unknown session is not undefined');
done();
});
});
it('it should only contain one session', function(done) {
this.memStore.length(function(err, len) {
should.not.exist(err, 'session count returned an error');
should.exist(len);
len.should.equal(1);
done();
});
});
it('it should clear all session records', function(done) {
var that = this;
this.memStore.clear(function(err, success) {
should.not.exist(err, 'clear returned an error');
success.should.be.true;
that.memStore.length(function(err, len) {
should.not.exist(err, 'session count after clear returned an error');
should.exist(len);
len.should.equal(0);
done();
});
});
});
it('it should destroy a session', function(done) {
var that = this;
this.memStore.set('555666777', {cookie: {maxAge:1000}, name: 'Rob Dobilina'}, function(err, rows) {
should.not.exist(err, 'set() returned an error');
rows.should.be.empty;
that.memStore.destroy('555666777', function(err) {
should.not.exist(err, 'destroy returned an error');
that.memStore.length(function(err, len) {
should.not.exist(err, 'session count after destroy returned an error');
should.exist(len);
len.should.equal(0);
done();
});
});
});
});
});
describe('connect-sqlite3 shared cache', function() {
it("should retrieve in one cached session what's stored in another.", function(done) {
var cwd = process.cwd();
var memStore = new SQLiteStore({db: 'file::memory:?cache=shared', mode: 0x20046});
process.chdir('..'); // Ensure we aren't opening a shared disk file
var memStore2 = new SQLiteStore({db: 'file::memory:?cache=shared', mode: 0x20046});
memStore.set('1111222233334444', {cookie: {maxAge:2011}, name: 'sample name'}, function(err, rows) {
process.chdir(cwd); // Restore dir
should.not.exist(err, 'set() returned an error');
rows.should.be.empty;
memStore2.get('1111222233334444', function(err, session) {
should.not.exist(err, 'get() returned an error');
should.exist(session);
(session).should.eql({cookie: {maxAge:2011}, name: 'sample name'});
done();
});
});
});
it("should not retrieve in one uncached session what's stored in another.", function(done) {
var memStore = new SQLiteStore({db: ':memory:'});
var memStore2 = new SQLiteStore({db: ':memory:'});
memStore.set('1111222233334444', {cookie: {maxAge:2011}, name: 'sample name'}, function(err, rows) {
should.not.exist(err, 'set() returned an error');
rows.should.be.empty;
memStore2.get('1111222233334444', function(err, session) {
should.not.exist(err, 'get() returned an error');
should.not.exist(session);
done();
});
});
});
});

16
server/node_modules/connect-sqlite3/test/tests.html generated vendored Normal file
View File

@@ -0,0 +1,16 @@
<html>
<head>
<title>Mocha</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<link rel="stylesheet" href="mocha.css" />
</head>
<body>
<div id="mocha"></div>
<script src="mocha.js"></script>
<script>mocha.setup('bdd')</script>
<script src="tests.js"></script>
<script>
mocha.run();
</script>
</body>
</html>