Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

new alerts front end #1846

Closed
wants to merge 51 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
c0efe19
add mariadb-client to Docker packages
struan Oct 9, 2024
716d068
add categorised alerts to user data
struan Oct 9, 2024
7b9743c
optionally return alert parts from prettify and add parts to alerts
struan Oct 9, 2024
4ece5df
update alerts page to use new look for displaying alerts
lucascumsille Oct 1, 2024
db0494d
handle sections for alerts prettyCriteria
struan Oct 10, 2024
c861390
display which sections an alert is limited to
struan Oct 10, 2024
9fa20e4
new forms for editing keyword and speaker alerts
struan Oct 24, 2024
9b1fedc
fix some search tests
struan Oct 24, 2024
e52992b
add vector_search_suggestions table
struan Oct 28, 2024
09528b1
show suggested search terms when creating an alert
struan Oct 28, 2024
761080e
script to import search suggestions into the database
struan Oct 29, 2024
1c0311e
make new alert searches use OR for keywords
struan Oct 28, 2024
9e728ec
optionally enable matching all terms in an alert
struan Oct 28, 2024
70b1bbb
group MP alerts together on alert page
struan Oct 28, 2024
fe96373
do not display disambigation options after editing an alert
struan Oct 29, 2024
80d7624
only use alertsearch on alert page for queries from elsewhere
struan Oct 29, 2024
84f35cb
display your own MP's keywords alerts even if no speaks alert
struan Oct 31, 2024
e27eb1b
add in some more alert page tests
struan Oct 29, 2024
251f82c
Tidy up front-end
lucascumsille Nov 12, 2024
e83ea9d
Remove discard changes buttton
lucascumsille Nov 12, 2024
6020246
fixup! Tidy up front-end
lucascumsille Nov 12, 2024
99b8cb0
Added minor improvements to buttons
lucascumsille Nov 12, 2024
9d0db9d
fixup! Tidy up front-end
lucascumsille Nov 12, 2024
d5e75e5
add some diagnostics to the vector search import script
struan Nov 13, 2024
f017fcb
spelling fix
struan Nov 14, 2024
ec3a379
fixup! Tidy up front-end
lucascumsille Nov 18, 2024
954522e
Added frontend to MP section
lucascumsille Nov 18, 2024
2d70cf1
fixup! Tidy up front-end
lucascumsille Nov 18, 2024
98e3260
fix red border on abandon changed button
struan Dec 3, 2024
ad2000e
add abandon changes button to first step of alert wizard
struan Dec 3, 2024
54344b9
Moved button icons from right to left
lucascumsille Dec 3, 2024
70371c8
fixup! Moved button icons from right to left
lucascumsille Dec 3, 2024
5830227
Improved indent for accordion items
lucascumsille Dec 3, 2024
14fa6b7
improve suggested terms workflow
struan Dec 3, 2024
d5d1c38
select and disable all suggested terms if click add all
struan Dec 3, 2024
f37be22
Moved search tips to alert form
lucascumsille Dec 3, 2024
9ab7168
add abandon changes button to suggestions step
struan Dec 3, 2024
f8618d3
small improvements to own mp alert section
struan Dec 3, 2024
232c65d
better detection of MP related alerts for grouping
struan Dec 3, 2024
1f20bc3
Reduced the amount of colour in alert page
lucascumsille Dec 4, 2024
d9b60c1
Improve message front-end when user is not subscribed to MP
lucascumsille Dec 4, 2024
fb7d719
Improve layout for alert-page-header
lucascumsille Dec 4, 2024
0100cfc
Remove alert-mockup file
lucascumsille Dec 4, 2024
52b971f
Improve color for open accordion, so it's easier to scan alert page
lucascumsille Dec 4, 2024
52521db
fixup! better detection of MP related alerts for grouping
struan Dec 4, 2024
5638cfa
fixup! fixup! better detection of MP related alerts for grouping
struan Dec 4, 2024
1d3d075
fix alert criteria parsing if only a single quoted phrase
struan Dec 4, 2024
75679ba
fix delete all button on alerts page
struan Dec 4, 2024
0004aca
fixup! fixup! fixup! better detection of MP related alerts for grouping
struan Dec 4, 2024
d306bbf
fixup! fix delete all button on alerts page
struan Dec 4, 2024
fba24b9
show all reps if using postcode to signup to MP alert
struan Dec 4, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ RUN apt-get -qq update && apt-get -qq install \
php-xdebug \
gettext \
rsync \
mariadb-client \
--no-install-recommends && \
rm -r /var/lib/apt/lists/*

Expand Down
389 changes: 367 additions & 22 deletions classes/AlertView/Standard.php

Large diffs are not rendered by default.

85 changes: 81 additions & 4 deletions classes/Utility/Alert.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,27 @@
*/

class Alert {
#XXX don't calculate this every time
public static function sectionToTitle($section) {
$section_map = [
"uk" => gettext('All UK'),
"debates" => gettext('House of Commons debates'),
"whalls" => gettext('Westminster Hall debates'),
"lords" => gettext('House of Lords debates'),
"wrans" => gettext('Written answers'),
"wms" => gettext('Written ministerial statements'),
"standing" => gettext('Bill Committees'),
"future" => gettext('Future Business'),
"ni" => gettext('Northern Ireland Assembly Debates'),
"scotland" => gettext('All Scotland'),
"sp" => gettext('Scottish Parliament Debates'),
"spwrans" => gettext('Scottish Parliament Written answers'),
"wales" => gettext('Welsh parliament record'),
"lmqs" => gettext('Questions to the Mayor of London'),
];

return $section_map[$section];
}
public static function detailsToCriteria($details) {
$criteria = [];

Expand All @@ -20,6 +41,10 @@ public static function detailsToCriteria($details) {
$criteria[] = 'speaker:' . $details['pid'];
}

if (!empty($details['search_section'])) {
$criteria[] = 'section:' . $details['search_section'];
}

$criteria = join(' ', $criteria);
return $criteria;
}
Expand All @@ -34,6 +59,7 @@ public static function forUser($email) {
$alerts = [];
foreach ($q as $row) {
$criteria = self::prettifyCriteria($row['criteria']);
$parts = self::prettifyCriteria($row['criteria'], true);
$token = $row['alert_id'] . '-' . $row['registrationtoken'];

$status = 'confirmed';
Expand All @@ -43,36 +69,87 @@ public static function forUser($email) {
$status = 'suspended';
}

$alerts[] = [
$alert = [
'token' => $token,
'status' => $status,
'criteria' => $criteria,
'raw' => $row['criteria'],
'keywords' => [],
'exclusions' => [],
'sections' => [],
];

$alert = array_merge($alert, $parts);

$alerts[] = $alert;
}

return $alerts;
}

public static function prettifyCriteria($alert_criteria) {
public static function prettifyCriteria($alert_criteria, $as_parts = false) {
$text = '';
$parts = ['words' => [], 'sections' => [], 'exclusions' => [], 'match_all' => true];
if ($alert_criteria) {
$criteria = explode(' ', $alert_criteria);
# check for phrases
if (strpos($alert_criteria, ' OR ') !== false) {
$parts['match_all'] = false;
}
$alert_criteria = str_replace(' OR ', ' ', $alert_criteria);
$alert_criteria = str_replace(['(', ')'], '', $alert_criteria);
if (strpos($alert_criteria, '"') !== false) {
# match phrases
preg_match_all('/"([^"]*)"/', $alert_criteria, $phrases);
# and then remove them from the criteria
$alert_criteria = trim(preg_replace('/ +/', ' ', str_replace($phrases[0], "", $alert_criteria)));

# and then create an array with the words and phrases
$criteria = [];
if ( $alert_criteria != "") {
$criteria = explode(' ', $alert_criteria);
}
$criteria = array_merge($criteria, $phrases[1]);
} else {
$criteria = explode(' ', $alert_criteria);
}
$words = [];
$exclusions = [];
$sections = [];
$sections_verbose = [];
$spokenby = array_values(\MySociety\TheyWorkForYou\Utility\Search::speakerNamesForIDs($alert_criteria));

foreach ($criteria as $c) {
if (!preg_match('#^speaker:(\d+)#', $c, $m)) {
if (preg_match('#^section:(\w+)#', $c, $m)) {
$sections[] = $m[1];
$sections_verbose[] = self::sectionToTitle($m[1]);
} elseif (strpos($c, '-') === 0) {
$exclusions[] = str_replace('-', '', $c);
} elseif (!preg_match('#^speaker:(\d+)#', $c, $m)) {
$words[] = $c;
}
}
if ($spokenby && count($words)) {
$text = implode(' or ', $spokenby) . ' mentions [' . implode(' ', $words) . ']';
$parts['spokenby'] = $spokenby;
$parts['words'] = $words;
} elseif (count($words)) {
$text = '[' . implode(' ', $words) . ']' . ' is mentioned';
$parts['words'] = $words;
} elseif ($spokenby) {
$text = implode(' or ', $spokenby) . " speaks";
$parts['spokenby'] = $spokenby;
}

if ($sections) {
$text = $text . " in " . implode(' or ', $sections_verbose);
$parts['sections'] = $sections;
$parts['sections_verbose'] = $sections_verbose;
}

$parts['exclusions'] = $exclusions;
}
if ($as_parts) {
return $parts;
}
return $text;
}
Expand Down
37 changes: 33 additions & 4 deletions classes/Utility/Search.php
Original file line number Diff line number Diff line change
Expand Up @@ -249,12 +249,19 @@ public static function searchMemberDbLookupWithNames($searchstring, $current_onl
* saying whether it was a postcode used.
*/

public static function searchConstituenciesByQuery($searchterm) {
public static function searchConstituenciesByQuery($searchterm, $mp_only=true) {
if (validate_postcode($searchterm)) {
// Looks like a postcode - can we find the constituency?
$constituency = Postcode::postcodeToConstituency($searchterm);
if ($constituency) {
return [ [$constituency], true ];
if ($mp_only) {
$constituency = Postcode::postcodeToConstituency($searchterm);
if ($constituency) {
return [ [$constituency], true ];
}
} else {
$constituencies = Postcode::postcodeToConstituencies($searchterm);
if ($constituencies) {
return [ $constituencies, true ];
}
}
}

Expand Down Expand Up @@ -297,6 +304,28 @@ public static function speakerNamesForIDs($searchstring) {
return $speakers;
}

/**
* get list of members of speaker IDs from search string
*
* @param string $searchstring The search string with the speaker:NNN text
*
* @return array Array with the speaker id string as key and speaker name as value
*/

public static function membersForIDs($searchstring) {
$criteria = explode(' ', $searchstring);
$speakers = [];

foreach ($criteria as $c) {
if (preg_match('#^speaker:(\d+)#', $c, $m)) {
$MEMBER = new \MEMBER(['person_id' => $m[1]]);
$speakers[$m[1]] = $MEMBER;
}
}

return $speakers;
}

/**
* replace speaker:NNNN with speaker:Name in search string
*
Expand Down
5 changes: 5 additions & 0 deletions db/0025-add-vector-search-suggestions.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
CREATE TABLE `vector_search_suggestions` (
`search_term` varchar(100) NOT NULL default '',
`search_suggestion` varchar(100) NOT NULL default '',
KEY `search_term` (`search_term`)
);
6 changes: 6 additions & 0 deletions db/schema.sql
Original file line number Diff line number Diff line change
Expand Up @@ -214,6 +214,12 @@ CREATE TABLE `postcode_lookup` (
PRIMARY KEY (`postcode`)
);

CREATE TABLE `vector_search_suggestions` (
`search_term` varchar(100) NOT NULL default '',
`search_suggestion` varchar(100) NOT NULL default '',
KEY `search_term` (`search_term`)
);

-- each time we index, we increment the batch number;
-- can use this to speed up search
CREATE TABLE `indexbatch` (
Expand Down
140 changes: 140 additions & 0 deletions scripts/import_search_suggestions.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
#!/usr/bin/env python3
# encoding: utf-8
"""
import_search_suggestions.py - Import vector search suggestions

See python scripts/import_search_suggestions.py --help for usage.

"""

import re
import sys
from pathlib import Path
from typing import cast
from warnings import filterwarnings

import MySQLdb
import pandas as pd
import rich_click as click
from pylib.mysociety import config
from rich import print
from rich.prompt import Prompt

repository_path = Path(__file__).parent.parent

config.set_file(repository_path / "conf" / "general")

# suppress warnings about using mysqldb in pandas
filterwarnings(
"ignore",
category=UserWarning,
message=".*pandas only supports SQLAlchemy connectable.*",
)


@click.group()
def cli():
pass


def get_twfy_db_connection() -> MySQLdb.Connection:
db_connection = cast(
MySQLdb.Connection,
MySQLdb.connect(
host=config.get("TWFY_DB_HOST"),
db=config.get("TWFY_DB_NAME"),
user=config.get("TWFY_DB_USER"),
passwd=config.get("TWFY_DB_PASS"),
charset="utf8",
),
)
return db_connection


def df_to_db(df: pd.DataFrame, verbose: bool = False):
"""
add search suggestions to the database
"""
df = df.dropna(how="any")
db_connection = get_twfy_db_connection()

with db_connection.cursor() as cursor:
# just remove everything and re-insert it all rather than trying to update things
cursor.execute("DELETE FROM vector_search_suggestions")
insert_command = "INSERT INTO vector_search_suggestions (search_term, search_suggestion) VALUES (%s, %s)"
suggestion_data = [
(row["original_query"], row["match"]) for _, row in df.iterrows()
]
cursor.executemany(insert_command, suggestion_data)
db_connection.commit()

if verbose:
print(f"[green]{len(df)} rows updated.")

db_connection.close()


def url_to_db(url: str, verbose: bool = False):
"""
Pipe external URL into the update process.
"""
df = pd.read_csv(url)

df_to_db(df, verbose=verbose)


def file_to_db(file: str, verbose: bool = False):
"""
Pipe file into the update process.
"""
df = pd.read_csv(file)

df_to_db(df, verbose=verbose)


@cli.command()
@click.option(
"--url",
required=False,
default=None,
help="A csv file to update search suggestions from.",
)
@click.option(
"--file",
required=False,
default=None,
help="A csv file to update search suggestions from.",
)
@click.option("--verbose", is_flag=True, help="Show verbose output")
def update_vector_search_suggestions(url: str, file: str, verbose: bool = False):
"""
Update the vector search suggestions
"""
if file:
file_to_db(file, verbose=verbose)
elif url:
url_to_db(url, verbose=verbose)


@cli.command()
def count_suggestions():
"""
for diagnostics to check import has worked
"""
db_connection = get_twfy_db_connection()
with db_connection.cursor() as cursor:
cursor.execute(
"select count(*) as num_suggestions from vector_search_suggestions"
)
count = cursor.fetchone()[0]
print(f"There are {count} suggestions in the db")

db_connection.close()


def main():
cli()


if __name__ == "__main__":
main()
Loading
Loading