Skip to content

Commit 8d18135

Browse files
committed
Added subscribers demonstration.
1 parent b211266 commit 8d18135

File tree

21 files changed

+378
-19
lines changed

21 files changed

+378
-19
lines changed

src/infra/sequelize/config/config.js

-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,6 @@ module.exports.connection = new Sequelize(database, username, password, {
4545
port: 3306,
4646
dialectOptions: {
4747
multipleStatements: true,
48-
useUTC: true,
4948
},
5049
pool: {
5150
max: 5,

src/infra/sequelize/migrations/20190625131808-initial-migration.ts

+86-2
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,54 @@ import runner from '../runner'
33

44
export default {
55
up: (queryInterface, Sequelize) => {
6+
const CREATE_BASE_USER = () => (
7+
queryInterface.createTable('base_user', {
8+
base_user_id: {
9+
type: Sequelize.UUID,
10+
defaultValue: Sequelize.UUIDV4,
11+
allowNull: false,
12+
primaryKey: true
13+
},
14+
first_name: {
15+
type: Sequelize.STRING(250),
16+
allowNull: false
17+
},
18+
last_name: {
19+
type: Sequelize.STRING(250),
20+
allowNull: false
21+
},
22+
user_email: {
23+
type: Sequelize.STRING(250),
24+
allowNull: false,
25+
unique: true
26+
},
27+
user_password: {
28+
type: Sequelize.STRING,
29+
allowNull: true,
30+
defaultValue: null
31+
},
32+
is_email_verified: {
33+
type: Sequelize.BOOLEAN,
34+
allowNull: false,
35+
defaultValue: false
36+
},
37+
username: {
38+
type: Sequelize.STRING(250),
39+
allowNull: true
40+
},
41+
created_at: {
42+
type: Sequelize.DATE,
43+
allowNull: false,
44+
defaultValue: Sequelize.literal('CURRENT_TIMESTAMP'),
45+
},
46+
updated_at: {
47+
type: Sequelize.DATE,
48+
allowNull: false,
49+
defaultValue: Sequelize.literal('CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP')
50+
}
51+
})
52+
)
53+
654
const CREATE_ARTIST = () => (
755
queryInterface.createTable('artist', {
856
artist_id: {
@@ -63,16 +111,52 @@ export default {
63111
})
64112
)
65113

114+
const CREATE_TRADER = () => (
115+
queryInterface.createTable('trader', {
116+
trader_id: {
117+
type: Sequelize.UUID,
118+
defaultValue: Sequelize.UUIDV4,
119+
allowNull: false,
120+
primaryKey: true
121+
},
122+
trader_base_id: {
123+
type: Sequelize.UUID,
124+
allowNull: false,
125+
primaryKey: true,
126+
references: {
127+
model: 'base_user',
128+
key: 'base_user_id'
129+
},
130+
onDelete: 'cascade',
131+
onUpdate: 'cascade',
132+
},
133+
created_at: {
134+
type: Sequelize.DATE,
135+
allowNull: false,
136+
defaultValue: Sequelize.literal('CURRENT_TIMESTAMP'),
137+
},
138+
updated_at: {
139+
type: Sequelize.DATE,
140+
allowNull: false,
141+
defaultValue: Sequelize.literal('CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP')
142+
}
143+
})
144+
)
145+
66146
return runner.run([
147+
() => CREATE_BASE_USER(),
67148
() => CREATE_ARTIST(),
68-
() => CREATE_VINYL()
149+
() => CREATE_VINYL(),
150+
() => CREATE_TRADER()
69151
])
70152
},
71153

72154
down: (queryInterface, Sequelize) => {
73155
return runner.run([
156+
() => queryInterface.dropTable('artist'),
74157
() => queryInterface.dropTable('vinyl'),
75-
() => queryInterface.dropTable('artist')
158+
() => queryInterface.dropTable('trader'),
159+
() => queryInterface.dropTable('base_user')
76160
])
77161
}
78162
};

src/infra/sequelize/models/baseUser.js

+9
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,15 @@ module.exports = function(sequelize, DataTypes) {
2020
allowNull: false,
2121
unique: true
2222
},
23+
is_email_verified: {
24+
type: DataTypes.BOOLEAN,
25+
allowNull: false,
26+
defaultValue: false
27+
},
28+
username: {
29+
type: DataTypes.STRING(250),
30+
allowNull: true
31+
},
2332
user_password: {
2433
type: DataTypes.STRING,
2534
allowNull: true,

src/modules/.DS_Store

6 KB
Binary file not shown.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
2+
export type SlackChannel = 'growth' | 'support';

src/modules/notification/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
2+
import "./subscribers";
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
2+
import { SlackService } from "./slack";
3+
4+
const slackService = new SlackService();
5+
6+
export {
7+
slackService
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
2+
import axios from 'axios'
3+
import { SlackChannel } from '../../domain/slackChannel';
4+
5+
export interface ISlackService {
6+
sendMessage (text: string, channel: SlackChannel): Promise<any>
7+
}
8+
9+
export class SlackService implements ISlackService {
10+
private growthChannelHookUrl: string = 'https://hooks.slack.com/services/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';
11+
private supportChannelHookUrl: string = 'https://hooks.slack.com/services/THKgeessd/Beese26CQ/mI66effeggeJCNa8bFVOwyAS';
12+
13+
constructor () {
14+
15+
}
16+
17+
private getWebookUrl (channel: SlackChannel): string {
18+
switch (channel) {
19+
case 'growth':
20+
return this.growthChannelHookUrl;
21+
case 'support':
22+
return this.supportChannelHookUrl;
23+
default:
24+
return "";
25+
}
26+
}
27+
28+
sendMessage (text: string, channel: SlackChannel): Promise<any> {
29+
const url: string = this.getWebookUrl(channel);
30+
return axios.post(url, { text });
31+
}
32+
33+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
2+
import { IHandle } from "../../../core/domain/events/IHandle";
3+
import { DomainEvents } from "../../../core/domain/events/DomainEvents";
4+
import { UserCreatedEvent } from "../../users/domain/events/userCreatedEvent";
5+
import { NotifySlackChannel } from "../useCases/notifySlackChannel/NotifySlackChannel";
6+
import { User } from "../../users/domain/user";
7+
8+
export class AfterUserCreated implements IHandle<UserCreatedEvent> {
9+
private notifySlackChannel: NotifySlackChannel;
10+
11+
constructor (notifySlackChannel: NotifySlackChannel) {
12+
this.setupSubscriptions();
13+
this.notifySlackChannel = notifySlackChannel;
14+
}
15+
16+
setupSubscriptions(): void {
17+
DomainEvents.register(this.onUserCreatedEvent.bind(this), UserCreatedEvent.name);
18+
}
19+
20+
private craftSlackMessage (user: User): string {
21+
return `Hey! Guess who just joined us? => ${user.firstName} ${user.lastName}\n
22+
Need to reach 'em? Their email is ${user.email}.`
23+
}
24+
25+
private async onUserCreatedEvent (event: UserCreatedEvent): Promise<void> {
26+
const { user } = event;
27+
28+
try {
29+
await this.notifySlackChannel.execute({
30+
channel: 'growth',
31+
message: this.craftSlackMessage(user)
32+
})
33+
} catch (err) {
34+
35+
}
36+
}
37+
}

src/modules/notification/domain/subscription/vinylCreatedSubscription.ts src/modules/notification/subscribers/AfterVinylCreated.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11

2-
import { IHandle } from "../../../../core/domain/events/IHandle";
3-
import { VinylCreatedEvent } from "../../../vinyl/domain/events/vinylCreatedEvent";
4-
import { DomainEvents } from "../../../../core/domain/events/DomainEvents";
5-
import { IVinylRepo } from "../../../vinyl/repos/vinylRepo";
2+
import { IVinylRepo } from "../../vinyl/repos/vinylRepo";
3+
import { IHandle } from "../../../core/domain/events/IHandle";
4+
import { VinylCreatedEvent } from "../../vinyl/domain/events/vinylCreatedEvent";
5+
import { DomainEvents } from "../../../core/domain/events/DomainEvents";
66

7-
export class VinylCreatedSubscription implements IHandle<VinylCreatedEvent> {
7+
export class AfterVinylCreated implements IHandle<VinylCreatedEvent> {
88
private vinylRepo: IVinylRepo;
99

1010
constructor () {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
2+
import { AfterUserCreated } from "./AfterUserCreated";
3+
import { notifySlackChannel } from "../useCases/notifySlackChannel";
4+
import { AfterVinylCreated } from "./AfterVinylCreated";
5+
6+
// Subscribers
7+
new AfterUserCreated(notifySlackChannel);
8+
new AfterVinylCreated();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
2+
import { UseCase } from "../../../../core/domain/UseCase";
3+
import { SlackChannel } from "../../domain/slackChannel";
4+
import { ISlackService } from "../../services/slack";
5+
6+
interface Request {
7+
channel: SlackChannel;
8+
message: string;
9+
}
10+
11+
export class NotifySlackChannel implements UseCase<Request, Promise<void>> {
12+
private slackService: ISlackService;
13+
14+
constructor (slackService: ISlackService) {
15+
this.slackService = slackService;
16+
}
17+
18+
async execute (req: Request): Promise<void> {
19+
await this.slackService.sendMessage(req.message, req.channel);
20+
}
21+
}
22+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
2+
import { NotifySlackChannel } from "./NotifySlackChannel";
3+
import { slackService } from "../../services";
4+
5+
const notifySlackChannel = new NotifySlackChannel(slackService);
6+
7+
export {
8+
notifySlackChannel
9+
}

src/modules/users/domain/user.ts

+12-2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ interface UserProps {
1616
profilePicture?: string;
1717
googleId?: number;
1818
facebookId?: number;
19+
username?: string;
1920
}
2021

2122
export class User extends AggregateRoot<UserProps> {
@@ -59,6 +60,14 @@ export class User extends AggregateRoot<UserProps> {
5960
return this.props.facebookId;
6061
}
6162

63+
get username (): string {
64+
return this.props.username;
65+
}
66+
67+
set username (value: string) {
68+
this.props.username = value;
69+
}
70+
6271
private constructor (props: UserProps, id?: UniqueEntityID) {
6372
super(props, id);
6473
}
@@ -72,7 +81,7 @@ export class User extends AggregateRoot<UserProps> {
7281
}
7382

7483
public static create (props: UserProps, id?: UniqueEntityID): Result<User> {
75-
84+
7685
const guardedProps = [
7786
{ argument: props.firstName, argumentName: 'firstName' },
7887
{ argument: props.lastName, argumentName: 'lastName' },
@@ -92,7 +101,8 @@ export class User extends AggregateRoot<UserProps> {
92101

93102
else {
94103
const user = new User({
95-
...props
104+
...props,
105+
username: props.username ? props.username : '',
96106
}, id);
97107

98108
const idWasProvided = !!id;

src/modules/users/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
2+
import "./subscribers";

src/modules/users/mappers/UserMap.ts

+26-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
1+
12
import { Mapper } from "../../../core/infra/Mapper";
23
import { User } from "../domain/user";
4+
import { UniqueEntityID } from "../../../core/domain/UniqueEntityID";
5+
import { UserEmail } from "../domain/userEmail";
6+
import { UserPassword } from "../domain/userPassword";
37

48
export class UserMap extends Mapper<User> {
59

@@ -8,9 +12,29 @@ export class UserMap extends Mapper<User> {
812
base_user_id: user.id.toString(),
913
user_email: user.email.value,
1014
user_password: user.password.value,
11-
user_first_name: user.firstName,
12-
user_last_name: user.lastName
15+
first_name: user.firstName,
16+
last_name: user.lastName,
17+
is_email_verified: user.isEmailVerified,
18+
username: user.username
1319
}
1420
}
21+
22+
public static toDomain (raw: any): User {
23+
const userEmailOrError = UserEmail.create(raw.user_email);
24+
const userPasswordOrError = UserPassword.create(raw.user_password);
25+
26+
const userOrError = User.create({
27+
email: userEmailOrError.getValue(),
28+
password: userPasswordOrError.getValue(),
29+
firstName: raw.first_name,
30+
lastName: raw.last_name,
31+
isEmailVerified: raw.is_email_verified,
32+
username: raw.username
33+
}, new UniqueEntityID(raw.base_user_id))
34+
35+
userOrError.isFailure ? console.log(userOrError.error) : '';
36+
37+
return userOrError.isSuccess ? userOrError.getValue() : null;
38+
}
1539

1640
}

0 commit comments

Comments
 (0)