mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 01:56:46 +00:00
* ⚡ Declutter test logs * 🐛 Fix random passwords length * 🐛 Fix password hashing in test user creation * 🐛 Hash leftover password * ⚡ Improve error message for `compare` * ⚡ Restore `randomInvalidPassword` contant * ⚡ Mock Telemetry module to prevent `--forceExit` * 🔥 Remove unused imports * 🔥 Remove unused import * ⚡ Add util for configuring test SMTP * ⚡ Isolate user creation * 🔥 De-duplicate `createFullUser` * ⚡ Centralize hashing * 🔥 Remove superfluous arg * 🔥 Remove outdated comment * ⚡ Prioritize shared tables during trucation * 🧪 Add login tests * ⚡ Use token helper * ✏️ Improve naming * ⚡ Make `createMemberShell` consistent * 🔥 Remove unneeded helper * 🔥 De-duplicate `beforeEach` * ✏️ Improve naming * 🚚 Move `categorize` to utils * ✏️ Update comment * 🧪 Simplify test * 📘 Improve `User.password` type * ⚡ Silence logger * ⚡ Simplify condition * ⚡ Unhash password in payload * 🐛 Fix comparison against unhashed password * ⚡ Increase timeout for fake SMTP service * 🔥 Remove unneeded import * ⚡ Use `isNull()` * 🧪 Use `Promise.all()` in creds tests * 🧪 Use `Promise.all()` in me tests * 🧪 Use `Promise.all()` in owner tests * 🧪 Use `Promise.all()` in password tests * 🧪 Use `Promise.all()` in users tests * ⚡ Re-set cookie if UM disabled * 🔥 Remove repeated line * ⚡ Refactor out shared owner data * 🔥 Remove unneeded import * 🔥 Remove repeated lines * ⚡ Organize imports * ⚡ Reuse helper * 🚚 Rename tests to match routers * 🚚 Rename `createFullUser()` to `createUser()` * ⚡ Consolidate user shell creation * ⚡ Make hashing async * ⚡ Add email to user shell * ⚡ Optimize array building * 🛠 refactor user shell factory * 🐛 Fix MySQL tests * ⚡ Silence logger in other DBs Co-authored-by: Ben Hesseldieck <b.hesseldieck@gmail.com>
138 lines
3.4 KiB
TypeScript
138 lines
3.4 KiB
TypeScript
/* eslint-disable import/no-cycle */
|
|
import {
|
|
AfterLoad,
|
|
AfterUpdate,
|
|
BeforeUpdate,
|
|
Column,
|
|
ColumnOptions,
|
|
CreateDateColumn,
|
|
Entity,
|
|
Index,
|
|
OneToMany,
|
|
ManyToOne,
|
|
PrimaryGeneratedColumn,
|
|
UpdateDateColumn,
|
|
} from 'typeorm';
|
|
import { IsEmail, IsString, Length } from 'class-validator';
|
|
import config = require('../../../config');
|
|
import { DatabaseType, IPersonalizationSurveyAnswers } from '../..';
|
|
import { Role } from './Role';
|
|
import { SharedWorkflow } from './SharedWorkflow';
|
|
import { SharedCredentials } from './SharedCredentials';
|
|
import { NoXss } from '../utils/customValidators';
|
|
import { answersFormatter } from '../utils/transformers';
|
|
|
|
export const MIN_PASSWORD_LENGTH = 8;
|
|
|
|
export const MAX_PASSWORD_LENGTH = 64;
|
|
|
|
function resolveDataType(dataType: string) {
|
|
const dbType = config.get('database.type') as DatabaseType;
|
|
|
|
const typeMap: { [key in DatabaseType]: { [key: string]: string } } = {
|
|
sqlite: {
|
|
json: 'simple-json',
|
|
},
|
|
postgresdb: {
|
|
datetime: 'timestamptz',
|
|
},
|
|
mysqldb: {},
|
|
mariadb: {},
|
|
};
|
|
|
|
return typeMap[dbType][dataType] ?? dataType;
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
|
|
function getTimestampSyntax() {
|
|
const dbType = config.get('database.type') as DatabaseType;
|
|
|
|
const map: { [key in DatabaseType]: string } = {
|
|
sqlite: "STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')",
|
|
postgresdb: 'CURRENT_TIMESTAMP(3)',
|
|
mysqldb: 'CURRENT_TIMESTAMP(3)',
|
|
mariadb: 'CURRENT_TIMESTAMP(3)',
|
|
};
|
|
|
|
return map[dbType];
|
|
}
|
|
|
|
@Entity()
|
|
export class User {
|
|
@PrimaryGeneratedColumn('uuid')
|
|
id: string;
|
|
|
|
@Column({ length: 254, nullable: true })
|
|
@Index({ unique: true })
|
|
@IsEmail()
|
|
email: string;
|
|
|
|
@Column({ length: 32, nullable: true })
|
|
@NoXss()
|
|
@IsString({ message: 'First name must be of type string.' })
|
|
@Length(1, 32, { message: 'First name must be $constraint1 to $constraint2 characters long.' })
|
|
firstName: string;
|
|
|
|
@Column({ length: 32, nullable: true })
|
|
@NoXss()
|
|
@IsString({ message: 'Last name must be of type string.' })
|
|
@Length(1, 32, { message: 'Last name must be $constraint1 to $constraint2 characters long.' })
|
|
lastName: string;
|
|
|
|
@Column({ nullable: true })
|
|
@IsString({ message: 'Password must be of type string.' })
|
|
password: string;
|
|
|
|
@Column({ type: String, nullable: true })
|
|
resetPasswordToken?: string | null;
|
|
|
|
// Expiration timestamp saved in seconds
|
|
@Column({ type: Number, nullable: true })
|
|
resetPasswordTokenExpiration?: number | null;
|
|
|
|
@Column({
|
|
type: resolveDataType('json') as ColumnOptions['type'],
|
|
nullable: true,
|
|
transformer: answersFormatter,
|
|
})
|
|
personalizationAnswers: IPersonalizationSurveyAnswers | null;
|
|
|
|
@ManyToOne(() => Role, (role) => role.globalForUsers, {
|
|
cascade: true,
|
|
nullable: false,
|
|
})
|
|
globalRole: Role;
|
|
|
|
@OneToMany(() => SharedWorkflow, (sharedWorkflow) => sharedWorkflow.user)
|
|
sharedWorkflows: SharedWorkflow[];
|
|
|
|
@OneToMany(() => SharedCredentials, (sharedCredentials) => sharedCredentials.user)
|
|
sharedCredentials: SharedCredentials[];
|
|
|
|
@CreateDateColumn({ precision: 3, default: () => getTimestampSyntax() })
|
|
createdAt: Date;
|
|
|
|
@UpdateDateColumn({
|
|
precision: 3,
|
|
default: () => getTimestampSyntax(),
|
|
onUpdate: getTimestampSyntax(),
|
|
})
|
|
updatedAt: Date;
|
|
|
|
@BeforeUpdate()
|
|
setUpdateDate(): void {
|
|
this.updatedAt = new Date();
|
|
}
|
|
|
|
/**
|
|
* Whether the user is pending setup completion.
|
|
*/
|
|
isPending: boolean;
|
|
|
|
@AfterLoad()
|
|
@AfterUpdate()
|
|
computeIsPending(): void {
|
|
this.isPending = this.password == null;
|
|
}
|
|
}
|