1
- use crate :: domain:: { NewSubscriber , SubscriberEmail , SubscriberName } ;
2
- use crate :: email_client:: EmailClient ;
1
+ use crate :: {
2
+ domain:: { NewSubscriber , SubscriberEmail , SubscriberName } ,
3
+ email_client:: EmailClient ,
4
+ startup:: ApplicationBaseUrl ,
5
+ } ;
3
6
use actix_web:: { web, HttpResponse } ;
4
7
use chrono:: Utc ;
8
+ use rand:: { distributions:: Alphanumeric , thread_rng, Rng } ;
9
+
5
10
use sqlx:: types:: Uuid ;
6
- use sqlx:: PgPool ;
11
+ use sqlx:: { PgPool , Postgres , Transaction } ;
7
12
8
13
#[ allow( dead_code) ]
9
14
#[ derive( serde:: Deserialize ) ]
@@ -12,9 +17,19 @@ pub struct FormData {
12
17
name : String ,
13
18
}
14
19
20
+ impl TryFrom < FormData > for NewSubscriber {
21
+ type Error = String ;
22
+
23
+ fn try_from ( value : FormData ) -> Result < Self , Self :: Error > {
24
+ let name = SubscriberName :: parse ( value. name ) ?;
25
+ let email = SubscriberEmail :: parse ( value. email ) ?;
26
+ Ok ( Self { name, email } )
27
+ }
28
+ }
29
+
15
30
#[ tracing:: instrument(
16
31
name = "Adding a new subscriber" ,
17
- skip( form, pool, email_client) ,
32
+ skip( form, pool, email_client, base_url ) ,
18
33
fields(
19
34
email = %form. email,
20
35
name = %form. name
@@ -24,61 +39,124 @@ pub async fn subscribe(
24
39
form : web:: Form < FormData > ,
25
40
pool : web:: Data < PgPool > ,
26
41
email_client : web:: Data < EmailClient > ,
42
+ base_url : web:: Data < ApplicationBaseUrl > ,
27
43
) -> HttpResponse {
28
44
let new_subscriber = match form. 0 . try_into ( ) {
29
45
Ok ( subscriber) => subscriber,
30
46
Err ( e) => return HttpResponse :: BadRequest ( ) . body ( e) ,
31
47
} ;
32
48
33
- if insert_subscriber ( & new_subscriber, & pool) . await . is_err ( ) {
49
+ let mut transaction = match pool. begin ( ) . await {
50
+ Ok ( t) => t,
51
+ Err ( _) => return HttpResponse :: InternalServerError ( ) . finish ( ) ,
52
+ } ;
53
+
54
+ let subscriber_id = match insert_subscriber ( & new_subscriber, & mut transaction) . await {
55
+ Ok ( id) => id,
56
+ Err ( _) => return HttpResponse :: InternalServerError ( ) . finish ( ) ,
57
+ } ;
58
+
59
+ let subscription_token = generate_subscription_token ( ) ;
60
+
61
+ if store_token ( subscriber_id, & subscription_token, & pool) . await . is_err ( ) {
34
62
return HttpResponse :: InternalServerError ( ) . finish ( ) ;
35
63
}
36
64
37
- if email_client
38
- . send_email (
39
- new_subscriber. email ,
40
- "Welcome!" ,
41
- "Welcome to our newsletter!" ,
42
- "Welcome to our newsletter!" ,
43
- )
65
+ if send_confirmation_email ( & email_client, new_subscriber, & base_url. 0 , & subscription_token)
44
66
. await
45
67
. is_err ( )
46
68
{
47
69
return HttpResponse :: InternalServerError ( ) . finish ( ) ;
48
70
}
49
71
72
+ if transaction. commit ( ) . await . is_err ( ) {
73
+ return HttpResponse :: InternalServerError ( ) . finish ( ) ;
74
+ }
75
+
50
76
HttpResponse :: Ok ( ) . finish ( )
51
77
}
52
78
53
- impl TryFrom < FormData > for NewSubscriber {
54
- type Error = String ;
79
+ #[ tracing:: instrument(
80
+ name = "Send a confirmation email to a new subscriber" ,
81
+ skip( new_subscriber, email_client)
82
+ ) ]
83
+ pub async fn send_confirmation_email (
84
+ email_client : & EmailClient ,
85
+ new_subscriber : NewSubscriber ,
86
+ base_url : & str ,
87
+ subscription_token : & str ,
88
+ ) -> Result < ( ) , reqwest:: Error > {
89
+ let confirmation_link = format ! (
90
+ "{}/subscriptions/confirm?subscription_token={}" ,
91
+ base_url, subscription_token
92
+ ) ;
93
+ let html_body_text = format ! (
94
+ "Welcome to our newsletter!<br />\
95
+ Click <a href=\" {}\" >here</a> to confirm your subscription.",
96
+ confirmation_link
97
+ ) ;
55
98
56
- fn try_from ( value : FormData ) -> Result < Self , Self :: Error > {
57
- let name = SubscriberName :: parse ( value. name ) ?;
58
- let email = SubscriberEmail :: parse ( value. email ) ?;
59
- Ok ( Self { name, email } )
60
- }
99
+ let plain_body_text = format ! (
100
+ "Welcome to our newsletter!\n Visit {} to confirm your subscription." ,
101
+ confirmation_link
102
+ ) ;
103
+
104
+ email_client
105
+ . send_email (
106
+ new_subscriber. email ,
107
+ "Welcome!" ,
108
+ & html_body_text,
109
+ & plain_body_text,
110
+ )
111
+ . await
61
112
}
62
113
63
114
#[ tracing:: instrument(
64
115
name = "Saving a new subscriber details in the database" ,
65
- skip( new_subscriber, pool )
116
+ skip( new_subscriber, transaction ) ,
66
117
) ]
67
118
async fn insert_subscriber (
68
119
new_subscriber : & NewSubscriber ,
69
- pool : & PgPool ,
70
- ) -> Result < ( ) , sqlx:: Error > {
71
- let subscribe_id = Uuid :: new_v4 ( ) ;
120
+ transaction : & mut Transaction < ' _ , Postgres >
121
+ ) -> Result < Uuid , sqlx:: Error > {
122
+ let subscriber_id = Uuid :: new_v4 ( ) ;
72
123
sqlx:: query!(
73
124
r#"
74
125
INSERT INTO subscriptions (id, email, name, subscribed_at, status)
75
- VALUES ($1, $2, $3, $4, 'confirmed ')
126
+ VALUES ($1, $2, $3, $4, 'pending_confirmation ')
76
127
"# ,
77
- subscribe_id ,
128
+ subscriber_id ,
78
129
new_subscriber. email. as_ref( ) ,
79
130
new_subscriber. name. as_ref( ) ,
80
131
Utc :: now( )
81
132
)
133
+ . execute ( transaction)
134
+ . await
135
+ . map_err ( |e| {
136
+ tracing:: error!( "Failed to execute query: {:?}" , e) ;
137
+ e
138
+ } ) ?;
139
+ Ok ( subscriber_id)
140
+ }
141
+
142
+ /// Store subscription token in the database
143
+ #[ tracing:: instrument(
144
+ name = "Saving subscription token in the database" ,
145
+ skip( subscriber_id, subscription_token, pool)
146
+ ) ]
147
+ async fn store_token (
148
+ subscriber_id : Uuid ,
149
+ subscription_token : & str ,
150
+ pool : & PgPool ,
151
+ ) -> Result < ( ) , sqlx:: Error > {
152
+ sqlx:: query!(
153
+ r#"
154
+ INSERT INTO subscription_tokens (subscription_token, subscriber_id)
155
+ VALUES ($1, $2)
156
+ "# ,
157
+ subscription_token,
158
+ subscriber_id,
159
+ )
82
160
. execute ( pool)
83
161
. await
84
162
. map_err ( |e| {
@@ -87,3 +165,12 @@ async fn insert_subscriber(
87
165
} ) ?;
88
166
Ok ( ( ) )
89
167
}
168
+
169
+ /// Gnerate a random 25 character-long case-sensitive subscription token
170
+ fn generate_subscription_token ( ) -> String {
171
+ let mut rng = thread_rng ( ) ;
172
+ std:: iter:: repeat_with ( || rng. sample ( Alphanumeric ) )
173
+ . map ( char:: from)
174
+ . take ( 25 )
175
+ . collect ( )
176
+ }
0 commit comments