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

Treat as same site #68

Open
wants to merge 2 commits into
base: future
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
77 changes: 60 additions & 17 deletions ext/webextension/src/browser_action/main_popup.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@

import {SiteStore} from "../lib/sitestore.js";
import {Site} from "../lib/sites.js";
import {defer, copy_to_clipboard} from "../lib/utils.js";
import {defer, copy_to_clipboard, regexpEscape} from "../lib/utils.js";
import {parseUri} from "../lib/uritools.js";
import config from "../lib/config.js";
import {ui} from "./ui.js";
Expand Down Expand Up @@ -241,21 +241,54 @@ function updateUIForDomainSettings(combined)
}

function extractDomainFromUrl(url) {
if (!url || url.startsWith('about:')
|| url.startsWith('resource:')
|| url.startsWith('moz-extension:')
|| url.startsWith('chrome-extension:')
|| url.startsWith('chrome:'))
url = '';
let domain_parts = parseUri(url).domain.split(".");
let significant_parts = 2;
const common_slds = ['co','com','gov','govt','net','org','edu','priv','ac'];
let second_level_domain = domain_parts[domain_parts.length-2].toLowerCase();
if (domain_parts.length > 2 && common_slds.includes(second_level_domain))
significant_parts = 3;

let significant_domain = domain_parts.slice(-significant_parts)
return [domain_parts, significant_domain];
if (!url)
throw "extractDomainFromUrl: No url provided!";
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Throwing here is not such a great Idea.. everything that follows extractDomainFromUrl will be skipped, in particular loadSites and updateUIForDomainSettings.
So instead of having access to all your sites, it appears to have lost all data.

probably better to return some sane default


var urlParsed = parseUri(url);
{
let protocolsIgnored = ["about", "resource", "moz-extension", "chrome-extension", "chrome", "edge", "brave"];
if (protocolsIgnored.includes(urlParsed.protocol))
throw "extractDomainFromUrl: Invalid url protocol provided";
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Especially here. about:blank is a perfectly legal page to open to just quickly lookup a password.

}

let rxDomainMatchers = [
/([^\.]+\.[^\.]+)$/i, // Default matcher matches two-part domain names (.* pattern) i.e. github.com
];

config.treat_as_same_site.split('\n').forEach(function(pattern) {
Copy link
Owner

@ttyridal ttyridal Nov 24, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I know I'm often guilty of this myself, but hey, now I'm in review mode and trying to understand someone else's code ;)

Could it be better to make it a named function and do something like
let matchRegEx = config.treat_as_same_site.split('\n').map(translate_match_string_to_regex); rxDomainMatchers.push(...matchRegEx.map(pattern => new RegExp(pattern, 'i')))

pattern = pattern.trim();
if (pattern.length == 0) return;
if (!pattern.startsWith('.')) return;
if (pattern.endsWith('.')) return;
//FIXME: More strict domain character parsing
// (should actually be config done at entry time)
pattern = '*' + pattern;
pattern = regexpEscape(pattern);
pattern = pattern.replace(/\\\*/g, '[^\.]+');
rxDomainMatchers.push(new RegExp(pattern, 'i'));
});

let domainMatched = null;

rxDomainMatchers.forEach(function(rx) {
let rxResult = rx.exec(urlParsed.domain);
if (!rxResult) return;

if (domainMatched == null)
return domainMatched = rxResult[0];

// Select domain with most parts that match
let domainMatchedParts = domainMatched.split('.');
let domainThisParts = rxResult[0].split('.');

if (domainThisParts.length > domainMatchedParts.length)
domainMatched = rxResult[0];
});

if (domainMatched == null)
throw 'extractDomainFromUrl: Could not match any sites! Check your configuration settings!';
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same thing about throw


return [urlParsed.domain.split('.'), domainMatched.split('.')];
}

function showSessionSetup() {
Expand Down Expand Up @@ -301,7 +334,17 @@ function popup(masterkey) {
}

window.addEventListener('load', function () {
config.get(['username', 'key_id', 'defaulttype', 'pass_to_clipboard', 'pass_store', 'passwdtimeout', 'use_sync', 'defaultname'])
config.get([
'username',
'key_id',
'defaulttype',
'pass_to_clipboard',
'pass_store',
'passwdtimeout',
'use_sync',
'defaultname',
'treat_as_same_site',
])
.then(v=>{
return runtimeSendMessage({action: 'masterkey_get', use_pass_store: !!v.pass_store});
})
Expand Down
32 changes: 29 additions & 3 deletions ext/webextension/src/css/mpwd.css
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,11 @@ button:focus, input:focus, mp-combobox:focus-within {
outline-style: auto;
}

input, select, mp-combobox {
input,
select,
mp-combobox,
textarea,
no-op {
background: #1b1d23;
color: #d7dae0;
height: 2em;
Expand Down Expand Up @@ -198,7 +202,8 @@ button#siteconfig_show {
}
#siteconfig > input,
#siteconfig > select,
#siteconfig > option {
#siteconfig > option,
no-op {
font-size: 1em;
font-weight: normal;
width: 6em;
Expand Down Expand Up @@ -302,7 +307,9 @@ button#siteconfig_show {

.configitem > input,
.configitem > select,
.configitem > option {
.configitem > option,
.configitem > textarea,
no-op {
margin-top:1.5em;
margin-left:-8em;
font-size: 1em;
Expand All @@ -316,6 +323,25 @@ button#siteconfig_show {
margin-top:0.2em;
}

.configitem > textarea#treat_as_same_site {
width: 11.5em;
height: 13em;
font-size: 0.7em;
margin-left: -11.5em;
margin-top: 2em;
white-space: nowrap;
}

/* FIXME: I don't understand how the left alignment of inputs works here,
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thats alright :) makes me remember I want to fix this cleanly with flexbox (which was very new back in 2015)

using negative margins and wrapping - so I just manually margin-left
by eye to line up - however it would be better to align this p
the same way as the input it is annotating */
.configitem > textarea#treat_as_same_site + p {
font-size: 0.6em;
margin-left: 6.7em;
line-height: normal;
}

#stored_sites {
width:100%;
text-align: center;
Expand Down
5 changes: 4 additions & 1 deletion ext/webextension/src/lib/config.js
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,9 @@ class Config {
get passwdtimeout() { if (typeof this._cache.passwdtimeout === 'undefined')
throw new Error("need get(['passwdtimeout'])");
else return this._cache.passwdtimeout; }
get treat_as_same_site() { if (typeof this._cache.treat_as_same_site === 'undefined')
throw new Error("need get(['treat_as_same_site'])");
else return this._cache.treat_as_same_site; }
get use_sync() { if (typeof this._cache.use_sync === 'undefined')
throw new Error("need get(['use_sync'])");
else return this._cache.use_sync; }
Expand Down Expand Up @@ -137,7 +140,7 @@ class Config {
if (lst.includes('username')) result.username = result.username || '';
if (lst.includes('pass_store')) result.pass_store = !!result.pass_store;
if (lst.includes('passwdtimeout')) result.passwdtimeout = isNaN(result.passwdtimeout) ? -1 : result.passwdtimeout;

if (lst.includes('treat_as_same_site')) result.treat_as_same_site = result.treat_as_same_site || [ '.ac.*', '.co.*', '.com.*', '.edu.*', '.geek.*', '.gov.*', '.govt.*', '.net.*', '.org.*', '.school.*' ].join('\n');
Object.assign(this._cache, result);

return singlekey ? result[lst[0]] : result;
Expand Down
5 changes: 5 additions & 0 deletions ext/webextension/src/lib/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -46,3 +46,8 @@ export function copy_to_clipboard(mimetype, data) {
document.execCommand("Copy", false, null);
document.oncopy=null;
}

export function regexpEscape(string) {
// https://stackoverflow.com/a/6969486
return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string
}
23 changes: 15 additions & 8 deletions ext/webextension/src/options/globaloptions.js
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,9 @@ document.querySelector('#auto_submit_username').addEventListener('change', funct
document.querySelector('#pass_store').addEventListener('change', function() {
config.set({pass_store: this.checked});
});
document.querySelector('#treat_as_same_site').addEventListener('change', function() {
config.set({treat_as_same_site: this.value});
});
document.querySelector('#use_sync').addEventListener('change', async function() {
let oldstore = (this.checked?chrome.storage.local:chrome.storage.sync);
let newstore = (!this.checked?chrome.storage.local:chrome.storage.sync);
Expand Down Expand Up @@ -150,14 +153,17 @@ document.querySelector('#use_sync').addEventListener('change', async function()
});

window.addEventListener('load', function() {
config.get(['defaulttype',
'defaultname',
'passwdtimeout',
'pass_to_clipboard',
'auto_submit_pass',
'auto_submit_username',
'pass_store',
'use_sync'])
config.get([
'defaulttype',
'defaultname',
'passwdtimeout',
'pass_to_clipboard',
'auto_submit_pass',
'auto_submit_username',
'pass_store',
'treat_as_same_site',
'use_sync',
])
.then(data => {
data = Object.assign({defaulttype: 'l', passwdtimeout: 0, pass_to_clipboard: true,
defaultname: '',
Expand All @@ -170,6 +176,7 @@ window.addEventListener('load', function() {
document.querySelector('#auto_submit_pass').checked = data.auto_submit_pass;
document.querySelector('#auto_submit_username').checked = data.auto_submit_username;
document.querySelector('#pass_store').checked = data.pass_store;
document.querySelector('#treat_as_same_site').value = data.treat_as_same_site;
document.querySelector('#use_sync').checked = data.use_sync;
});
});
Expand Down
116 changes: 65 additions & 51 deletions ext/webextension/src/options/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -34,59 +34,73 @@
<h2 class="accordion_toggle">Global settings</h2>
<div class="configgroup">
<div class="configsubgroup left">
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="passwdtype">Default password type:</label>
<select class="w3-select w3-small w3-border" id="passwdtype">
<option value="x">Maximum
<option value="l">Long
<option value="m">Medium
<option value="b">Basic
<option value="s">Short
<option value="i">Pin
<option value="n">Name
<option value="p">Phrase
<option value="nx">Name [V]
<option value="px">Phrase [V]
</select>
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="passwdtimeout">Remember password:</label>
<select class="w3-select w3-small w3-border" id="passwdtimeout">
<option value="0">Never
<option value="5">5 minutes
<option value="15">15 minutes
<option value="30">30 minutes
<option value="60">1 hour
<option value="480">8 hours
<option value="-1">Until closed
</select>
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="defaultname">Default username:</label>
<input class="w3-select w3-small w3-border" id="defaultname">
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="passwdtype">Default password type:</label>
<select class="w3-select w3-small w3-border" id="passwdtype">
<option value="x">Maximum
<option value="l">Long
<option value="m">Medium
<option value="b">Basic
<option value="s">Short
<option value="i">Pin
<option value="n">Name
<option value="p">Phrase
<option value="nx">Name [V]
<option value="px">Phrase [V]
</select>
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="passwdtimeout">Remember password:</label>
<select class="w3-select w3-small w3-border" id="passwdtimeout">
<option value="0">Never
<option value="5">5 minutes
<option value="15">15 minutes
<option value="30">30 minutes
<option value="60">1 hour
<option value="480">8 hours
<option value="-1">Until closed
</select>
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="defaultname">Default username:</label>
<input class="w3-select w3-small w3-border" id="defaultname">
</div>
<div class="configitem">
<label class="w3-label w3-small w3-text-white" for="treat_as_same_site">Override as same site</label>
<textarea class="w3-label w3-small w3-text-white" id="treat_as_same_site"></textarea>
<p class="w3-label w3-small w3-text-white">
- Enter overrides above, one per line<br />
- lines must start with a . (period)<br />
- lines that cannot be parsed are ignored<br />
- <b>*</b> is wildcard, will match anything except . (period)<br />
- Example:<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<b>.net.*</b> will treat <i>x.net.nz</i> as a site,<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;instead of ignoring <i>x</i> and using <i>net.nz</i> as the site
</p>
</div>

</div>
<div class="configsubgroup right">
<div class="configitem">
<label for="pass_to_clipboard">Copy password to clipboard</label>
<input id="pass_to_clipboard" type="checkbox"/>
</div>
<div class="configitem">
<label for="auto_submit_pass">Submit after password inject</label>
<input id="auto_submit_pass" type="checkbox"/>
</div>
<div class="configitem">
<label for="auto_submit_username">Inject username</label>
<input id="auto_submit_username" type="checkbox"/>
</div>
<div class="configitem">
<label for="pass_store">Use OS' password store</label>
<input id="pass_store" type="checkbox"/>
</div>
<div class="configitem">
<label for="use_sync">Let browser sync site data</label>
<input id="use_sync" type="checkbox"/>
</div>
<div class="configitem">
<label for="pass_to_clipboard">Copy password to clipboard</label>
<input id="pass_to_clipboard" type="checkbox"/>
</div>
<div class="configitem">
<label for="auto_submit_pass">Submit after password inject</label>
<input id="auto_submit_pass" type="checkbox"/>
</div>
<div class="configitem">
<label for="auto_submit_username">Inject username</label>
<input id="auto_submit_username" type="checkbox"/>
</div>
<div class="configitem">
<label for="pass_store">Use OS' password store</label>
<input id="pass_store" type="checkbox"/>
</div>
<div class="configitem">
<label for="use_sync">Let browser sync site data</label>
<input id="use_sync" type="checkbox"/>
</div>
</div>
</div>
</div>
Expand Down
15 changes: 13 additions & 2 deletions ext/webextension/src/options/options.html
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@
div.item {
break-inside: avoid;
width:100%;
height:3em;
display:flex;
justify-content: space-between;
align-items: center;
Expand All @@ -26,7 +25,9 @@

input,
select,
option {
option,
textarea,
no-op {
display:block;
border: 0;
width: 14em;
Expand All @@ -36,6 +37,11 @@
input[type=checkbox] {
width:1em;
}
#treat_as_same_site {
height: 15em;
white-space: nowrap;
Copy link
Owner

@ttyridal ttyridal Nov 24, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the nowrap is the problem.. seems like pre would be the correct(?) choice
ref https://bugzilla.mozilla.org/show_bug.cgi?id=1137650

overflow: auto;
}

</style>
</head>
Expand Down Expand Up @@ -71,6 +77,11 @@
<label for="defaultname">Default username</label>
<input id="defaultname">
</div>
<div class="item">
<label for="treat_as_same_site">Override as same site</label>
<textarea id="treat_as_same_site"></textarea>

</div>
<div class="item">
<label for="pass_to_clipboard">Copy password to clipboard</label>
<input id="pass_to_clipboard" type="checkbox"/>
Expand Down