-
-
Notifications
You must be signed in to change notification settings - Fork 9
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
Enable/Disable analytics based on user consent #3467
Conversation
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## master #3467 +/- ##
==========================================
+ Coverage 74.53% 74.59% +0.06%
==========================================
Files 286 285 -1
Lines 11023 11039 +16
Branches 1345 1342 -3
==========================================
+ Hits 8216 8235 +19
- Misses 2420 2421 +1
+ Partials 387 383 -4
Flags with carried forward coverage won't be shown. Click here to find out more. ☔ View full report in Codecov by Sentry. |
878bac9
to
ed56311
Compare
ed56311
to
8307f70
Compare
8339cea
to
1217a27
Compare
1217a27
to
fffdc2e
Compare
0ad3b39
to
ce7b9dc
Compare
49efc4f
to
d7d1e61
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 5 of 12 files at r2, all commit messages.
Reviewable status: 5 of 12 files reviewed, 7 unresolved discussions (waiting on @andracc)
Backend/Models/User.cs
line 69 at r2 (raw file):
public string Username { get; set; } [BsonElement("otelConsent")]
Let's add /// <summary>
comments for these two consent bools, since it might be confusing what the difference is.
Backend.Tests/Otel/OtelKernelTests.cs
line 40 at r2 (raw file):
[Test] public void BuildersSetConsentAndSessionBaggageFromHeader()
How about just BuildersSetBaggageFromHeader
(and OnEndSetsTagsFromBaggage
below)?
Backend/Otel/OtelKernel.cs
line 108 at r2 (raw file):
public override async void OnEnd(Activity data) { var consentString = data?.GetBaggageItem("otelConsentBaggage");
Why is the ?
necessary on data?
throughout this function?
src/components/UserSettings/UserSettings.tsx
line 82 at r2 (raw file):
const show = (): void => setDisplayConsent(true); const handleConsentChange = (consentVal: boolean | undefined): void => {
This can be shortened to consentVal?: boolean
src/components/UserSettings/UserSettings.tsx
line 303 at r2 (raw file):
data-testid={UserSettingsIds.ButtonChangeConsent} id={UserSettingsIds.ButtonChangeConsent} onClick={show}
Since show
isn't used elsewhere, we can just use a lambda function here:
onClick={() => setDisplayConsent(true)}
src/backend/index.ts
line 60 at r2 (raw file):
? ((config.headers.otelConsent = true), (config.headers.sessionId = getSessionId())) : (config.headers.otelConsent = false);
This <...> ? ((<do one thing>), (<do another thing>)) : (<...>)
notation is compact and logically correct, but the presence of an ordered pair ((...), (...))
could be confusing. How about a slightly longer, more common and explicit
if (LocalStorage.getCurrentUser()?.otelConsent) {
config.headers.otelConsent = true;
config.headers.sessionId = getSessionId();
} else {
config.headers.otelConsent = false;
}
src/components/AnalyticsConsent/AnalyticsConsent.tsx
line 0 at r2 (raw file):
If this file is changed to src/components/AnalyticsConsent/index.tsx
and export function
changed to export default function
, then that will match our style elsewhere, and it can be imported with
import AnalyticsConsent from "components/AnalyticsConsent";
rather than
import { AnalyticsConsent } from "components/AnalyticsConsent/AnalyticsConsent";
src/components/AnalyticsConsent/AnalyticsConsent.tsx
line 45 at r2 (raw file):
<Typography variant="h6" style={{ color: "#1976d2", fontWeight: 600 }}
Use color: themeColors.primary
rather than using a hex code here.
src/components/AnalyticsConsent/AnalyticsConsent.tsx
line 46 at r2 (raw file):
variant="h6" style={{ color: "#1976d2", fontWeight: 600 }} color="primary"
This color="primary"
is unnecessary, being overridden by the manual color definition in the style.
src/components/AnalyticsConsent/AnalyticsConsent.tsx
line 70 at r2 (raw file):
<Grid item> <Button color="primary"
The color="primary"
and this and the next button are unnecessary because "primary" is the default color.
src/components/AnalyticsConsent/AnalyticsConsent.tsx
line 72 at r2 (raw file):
color="primary" onClick={acceptAnalytics} style={{
These button dimensions may be too small for some ui languages, so let's bump up to
style={{ height: 60, width: 155 }}
in both these buttons.
src/components/App/AppLoggedIn.tsx
line 58 at r2 (raw file):
async function handleConsentChange( otelConsent: boolean | undefined
This can be shortened to otelConsent?: boolean
.
src/components/App/AppLoggedIn.tsx
line 107 at r2 (raw file):
onChangeConsent={handleConsentChange} required ></AnalyticsConsent>
This can be shortened to
<AnalyticsConsent onChangeConsent={handleConsentChange} required />
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 2 of 7 files at r3, all commit messages.
Reviewable status: 7 of 12 files reviewed, 9 unresolved discussions (waiting on @andracc)
src/components/UserSettings/UserSettings.tsx
line 79 at r3 (raw file):
} const [displayConsent, setDisplayConsent] = useState(false);
Would be nice for this useState
to be up with all the rest.
src/components/UserSettings/UserSettings.tsx
line 299 at r3 (raw file):
)} </Typography> <Button
Between </Typography>
and <Button`, let's insert:
</Grid>
<Grid item>
By putting the Typography
and Button
components in their own Grid
items, the ? icon with the tooltip can show up to the right of the button.
src/components/UserSettings/UserSettings.tsx
line 311 at r3 (raw file):
onChangeConsent={handleConsentChange} required={false} ></AnalyticsConsent>
></AnalyticsConsent>
can be replaced with />
src/components/AnalyticsConsent/index.tsx
line 5 at r3 (raw file):
import { ReactElement } from "react"; import { useTranslation } from "react-i18next"; import { themeColors } from "types/theme";
Add a blank line between the external imports (ending with "react-i18next"
) and the internal imports.
src/components/AnalyticsConsent/index.tsx
line 8 at r3 (raw file):
interface ConsentProps { onChangeConsent: (consentVal: boolean | undefined) => void;
This can be (consentVal?: boolean)
src/components/AnalyticsConsent/index.tsx
line 24 at r3 (raw file):
const clickedAway = (): void => { props.onChangeConsent(undefined); };
These can be 1-liners:
const acceptAnalytics = (): void => props.onChangeConsent(true);
const rejectAnalytics = (): void => props.onChangeConsent(false);
const clickedAway = (): void => props.onChangeConsent(undefined);
src/components/AnalyticsConsent/index.tsx
line 35 at r3 (raw file):
onClose={!props.required ? clickedAway : undefined} > <div style={{ padding: 20 }}>
Rather than having a div
inside the Drawer
and around the Grid
, let's add PaperProps={{ style: { padding: 20 } }}
to the Drawer
's props.
src/components/AnalyticsConsent/index.tsx
line 53 at r3 (raw file):
<Typography variant="body1" align="left"
This Typography
inherits a left- or right- alight from the parent depending on whether the interface is in a ltr or rtl language, so you should remove the align="left"
.
src/components/AnalyticsConsent/index.tsx
line 75 at r3 (raw file):
width: 155, }} variant={"contained"}
This and the button below can be trimmed up to
style={{ height: 60, width: 155 }}
variant="contained"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewable status: 7 of 12 files reviewed, 15 unresolved discussions (waiting on @andracc)
Backend/Otel/OtelKernel.cs
line 108 at r2 (raw file):
Previously, imnasnainaec (D. Ror.) wrote…
Why is the
?
necessary ondata?
throughout this function?
Part of my puzzlement is that the function parameter Activity data
has no indication that it can be null.
Backend/Otel/OtelKernel.cs
line 38 at r3 (raw file):
internal static void TrackConsent(Activity activity, HttpRequest request) { var consent = request.Headers.TryGetValue("otelConsent", out var values) ? bool.TryParse(values.FirstOrDefault(), out bool _) : true;
The bool
of out bool _
is unnecessary.
And to break up a too-long line into something a bit more readable, I suggest:
var consent = request.Headers.TryGetValue("otelConsent", out var values)
? bool.TryParse(values.FirstOrDefault(), out _)
: true;
Backend/Otel/OtelKernel.cs
line 111 at r3 (raw file):
data?.AddTag("otelConsent", consentString); var consent = bool.TryParse(consentString, out bool value) ? value : false; if (consent)
These two lines can be condensed into:
if (bool.TryParse(consentString, out bool consent) && consent)
Backend.Tests/Otel/OtelKernelTests.cs
line 21 at r3 (raw file):
private const string OtelSessionIdKey = "sessionId"; private const string OtelConsentBaggageKey = "otelConsentBaggage"; private const string OtelSessionBaggageKey = "sessionBaggage";
I think these Otel*Key
consts should be defined in the OtelKernel
class so that they can be used both there and here.
src/components/AnalyticsConsent/index.tsx
line 29 at r3 (raw file):
return ( <div>
Is there a reason for this outside <div>
?
src/components/AnalyticsConsent/index.tsx
line 60 at r3 (raw file):
</Typography> </Grid> <Grid
This Grid
, even though it has the container
prop, should also have the item
prop, because it is a child of a <Grid container ...
and because it uses the xs
prop.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 5 of 11 files at r4, all commit messages.
Reviewable status: 8 of 15 files reviewed, 7 unresolved discussions (waiting on @andracc)
src/components/App/AppLoggedIn.tsx
line 58 at r2 (raw file):
Previously, imnasnainaec (D. Ror.) wrote…
This can be shortened to
otelConsent?: boolean
.
... and it will save a couple lines to do so.
Backend.Tests/Otel/OtelKernelTests.cs
line 0 at r4 (raw file):
Any test(s) we should add for "false"
consent?
src/components/UserSettings/UserSettings.tsx
line 310 at r4 (raw file):
</Button> </Grid> {displayConsent ? (
Since displayConsent
is a boolean
, we can change ?
to &&
and drop the : null
.
src/types/user.ts
line 18 at r4 (raw file):
token: "", isAdmin: false, answeredConsent: false,
I'm not sure why we need to set answeredConsent: false,
here and can't leave it undefined as with otelConsent
.
Backend/Models/User.cs
line 70 at r4 (raw file):
/// <summary> /// Is true if user accepts analytics, false otherwise.
There's a extra space after otherwise.
src/components/AnalyticsConsent/index.tsx
line 23 at r4 (raw file):
function ConsentButton(props: { decision: () => void;
I like this component, though since it's a button,onClick
would be a more expected prop name than decision
.
src/components/AnalyticsConsent/index.tsx
line 62 at r4 (raw file):
decision={acceptAnalytics} text={t("analyticsConsent.consentModal.acceptAllBtn")} ></ConsentButton>
These two ></ConsentButton>
can be />
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 5 of 10 files at r5, all commit messages.
Reviewable status: 10 of 15 files reviewed, 4 unresolved discussions (waiting on @andracc)
Backend/Models/User.cs
line 70 at r5 (raw file):
/// <summary> /// Is true if user accepts analytics, false otherwise.
/// Is false if user rejects analytics, true otherwise.
src/components/AnalyticsConsent/index.tsx
line 53 at r5 (raw file):
{t("analyticsConsent.consentModal.title")} </Typography> <Typography variant="body1" style={{ marginRight: 25 }} gutterBottom>
Let's find a way (e.g., with Grid spacing) to make this look nice without marginRight
, since that won't work for rtl layouts. If you need marginRight
for the right look, then replace it with
style={document.body.dir === "rtl" ? { marginLeft:25 } : { marginRight: 25 }}
(...better not to use if we don't have to.)
src/backend/index.ts
line 61 at r5 (raw file):
config.headers.analyticsOn = `${!!consent}`; } else { config.headers.analyticsOn = true;
How about using a formatted string in both cases (with${false}
and ${true}
), so we're explicit on the strings being sent?
Previously, imnasnainaec (D. Ror.) wrote…
be careful here. You've got a bug at the moment. To strip away the other bits:
this will result in bool consent = false;
bool.TryParse(headerValue, out consent); if the parsing fails then whatever |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 3 of 4 files at r6, all commit messages.
Reviewable status: 13 of 16 files reviewed, 3 unresolved discussions (waiting on @andracc)
Backend/Otel/OtelKernel.cs
line 38 at r3 (raw file):
Previously, hahn-kev (Kevin Hahn) wrote…
be careful here. You've got a bug at the moment. To strip away the other bits:
string headerValue = "false"; var consent = bool.TryParse(headerValue, out _);
this will result in
consent
being true because TryParse returns true if it was able to parse the value. You want something like this:bool consent = false; bool.TryParse(headerValue, out consent);if the parsing fails then whatever
consent
equals will not change, so you can set your default value there.
Ah, good catch. Thanks, @hahn-kev. So we could've had something like
var consent = true;
request.Headers.TryGetValue("otelConsent", out var values) && bool.TryParse(values.FirstOrDefault(), out consent)
Though it seems Andra's further efforts and review has rendered the TryParse
s unnecessary.
Backend/Otel/OtelKernel.cs
line 128 at r6 (raw file):
data.AddTag("city", location?.City); } data.AddTag(OtelSessionId, data?.GetBaggageItem("sessionBaggage"));
There's one more data?
left.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 1 of 10 files at r5, 1 of 4 files at r6, 3 of 3 files at r7, all commit messages.
Reviewable status: complete! all files reviewed, all discussions resolved (waiting on @andracc)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Reviewed 1 of 2 files at r8, 2 of 2 files at r9, all commit messages.
Reviewable status: complete! all files reviewed, all discussions resolved (waiting on @andracc)
Resolves #3410, tracking analytics consent via a user property rather than cookies.
Users who have not previously accepted or rejected analytics will be prompted to do so at login. Users can thereafter update their response in User Settings.
This change is