aboutsummaryrefslogtreecommitdiffstats
path: root/src/libserver
diff options
context:
space:
mode:
Diffstat (limited to 'src/libserver')
-rw-r--r--src/libserver/url.c53
1 files changed, 52 insertions, 1 deletions
diff --git a/src/libserver/url.c b/src/libserver/url.c
index 1342ae92c..55f4c30e0 100644
--- a/src/libserver/url.c
+++ b/src/libserver/url.c
@@ -1454,6 +1454,16 @@ url_email_start (struct url_callback_data *cb,
return TRUE;
}
+ else {
+ /* Just '@' */
+
+ /* Check if this match is a part of the previous mailto: email */
+ if (cb->last_at != NULL && cb->last_at == pos) {
+ cb->last_at = NULL;
+ }
+
+ return FALSE;
+ }
return FALSE;
}
@@ -1463,7 +1473,7 @@ url_email_end (struct url_callback_data *cb,
const gchar *pos,
url_match_t *match)
{
- const gchar *last = NULL;
+ const gchar *last = NULL, *c, *p;
struct http_parser_url u;
if (!match->prefix || match->prefix[0] == '\0') {
@@ -1484,6 +1494,47 @@ url_email_end (struct url_callback_data *cb,
return TRUE;
}
+ else {
+ /*
+ * Here we have just '@', so we need to find both start and end of the
+ * pattern
+ */
+ g_assert (*pos == '@');
+
+ if (pos >= cb->end - 2 || pos <= cb->begin + 1) {
+ /* Boundary violation */
+ return FALSE;
+ }
+
+ if (!g_ascii_isalnum (pos[1]) || !g_ascii_isalnum (*(pos - 1))) {
+ return FALSE;
+ }
+
+ c = pos - 1;
+ while (c > cb->begin && is_usersafe (*c)) {
+ c --;
+ }
+ /* Rewind to the first alphanumeric character */
+ while (c < pos && !g_ascii_isalnum (c)) {
+ c ++;
+ }
+
+ /* Find the end of email */
+ p = pos + 1;
+ while (p < cb->end && is_domain (*p)) {
+ p ++;
+ }
+ /* Rewind it again to avoid bad emails to be detected */
+ while (p > pos && !g_ascii_isalnum (*p)) {
+ p --;
+ }
+
+ if (p > c) {
+ match->m_begin = c;
+ match->m_len = p - c;
+ return TRUE;
+ }
+ }
return FALSE;
}