Refactor number parsing and validation
Refactor libphonenumber.validateNumber into libphonenumber.parseNumber, which encapsulates the try-catch pattern used in number parsing and returns an object of info about the input number rather tha throwing since we expect to get some invalid number inputs the user is typing. In the conversation model, * Separate phone number validation from search token updating. * Perform token update before save if the number was valid. * Stop storing unneeded number variants as conversation properties. // FREEBIE
This commit is contained in:
parent
7dd0fb70b5
commit
a258f1a66b
6 changed files with 105 additions and 51 deletions
|
@ -34,20 +34,19 @@
|
||||||
return (cc !== 0) ? cc : "";
|
return (cc !== 0) ? cc : "";
|
||||||
},
|
},
|
||||||
|
|
||||||
verifyNumber: function(number, regionCode) {
|
parseNumber: function(number, defaultRegionCode) {
|
||||||
var parsedNumber = libphonenumber.parse(number, regionCode);
|
try {
|
||||||
|
var parsedNumber = libphonenumber.parse(number, defaultRegionCode);
|
||||||
|
|
||||||
if(!regionCode || regionCode == 'ZZ') {
|
return {
|
||||||
regionCode = libphonenumber.getRegionCodeForNumber(parsedNumber);
|
isValidNumber: libphonenumber.isValidNumber(parsedNumber),
|
||||||
}
|
regionCode: libphonenumber.getRegionCodeForNumber(parsedNumber),
|
||||||
|
countryCode: '' + parsedNumber.getCountryCode(),
|
||||||
var isValidNumber = libphonenumber.isValidNumber(parsedNumber);
|
nationalNumber: '' + parsedNumber.getNationalNumber(),
|
||||||
var isValidNumberForRegion = libphonenumber.isValidNumberForRegion(parsedNumber, regionCode);
|
e164: libphonenumber.format(parsedNumber, libphonenumber.PhoneNumberFormat.E164)
|
||||||
|
};
|
||||||
if (isValidNumber && isValidNumberForRegion) {
|
} catch (ex) {
|
||||||
return libphonenumber.format(parsedNumber, libphonenumber.PhoneNumberFormat.E164);
|
return { error: ex, isValidNumber: false };
|
||||||
} else {
|
|
||||||
throw new Error("The number seems not to be valid.");
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -42,7 +42,6 @@
|
||||||
conversation: this
|
conversation: this
|
||||||
});
|
});
|
||||||
|
|
||||||
this.on('change:id change:name', this.updateTokens);
|
|
||||||
this.on('change:avatar', this.updateAvatarUrl);
|
this.on('change:avatar', this.updateAvatarUrl);
|
||||||
this.on('destroy', this.revokeAvatarUrl);
|
this.on('destroy', this.revokeAvatarUrl);
|
||||||
},
|
},
|
||||||
|
@ -56,20 +55,20 @@
|
||||||
return "Invalid conversation type: " + attributes.type;
|
return "Invalid conversation type: " + attributes.type;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!attributes.tokens) {
|
var error = this.validateNumber();
|
||||||
return this.updateTokens();
|
if (error) { return error; }
|
||||||
}
|
|
||||||
|
this.updateTokens();
|
||||||
},
|
},
|
||||||
|
|
||||||
validateNumber: function() {
|
validateNumber: function() {
|
||||||
try {
|
if (this.isPrivate()) {
|
||||||
this.id = libphonenumber.util.verifyNumber(this.id);
|
var regionCode = storage.get('regionCode');
|
||||||
} catch (ex) {
|
var number = libphonenumber.util.parseNumber(this.id, regionCode);
|
||||||
if (ex === "Invalid country calling code") {
|
if (number.isValidNumber) {
|
||||||
var regionCode = storage.get('regionCode');
|
this.set({ id: number.e164 });
|
||||||
this.id = libphonenumber.util.verifyNumber(this.id, regionCode);
|
|
||||||
} else {
|
} else {
|
||||||
throw ex;
|
return number.error || "Invalid phone number";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -78,26 +77,17 @@
|
||||||
var tokens = [];
|
var tokens = [];
|
||||||
var name = this.get('name');
|
var name = this.get('name');
|
||||||
if (typeof name === 'string') {
|
if (typeof name === 'string') {
|
||||||
tokens = name.trim().toLowerCase().split(/[\s\-_\(\)\+]+/);
|
tokens.push(name.toLowerCase());
|
||||||
|
tokens = tokens.concat(name.trim().toLowerCase().split(/[\s\-_\(\)\+]+/));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.isPrivate()) {
|
if (this.isPrivate()) {
|
||||||
try {
|
var regionCode = storage.get('regionCode');
|
||||||
this.validateNumber();
|
var number = libphonenumber.util.parseNumber(this.id, regionCode);
|
||||||
var number = libphonenumber.util.splitCountryCode(this.id);
|
tokens.push(
|
||||||
var international_number = '' + number.country_code + number.national_number;
|
this.id,
|
||||||
var national_number = '' + number.national_number;
|
number.nationalNumber,
|
||||||
|
number.countryCode + number.nationalNumber
|
||||||
this.set({
|
);
|
||||||
id: this.id,
|
|
||||||
e164_number: this.id,
|
|
||||||
national_number: national_number,
|
|
||||||
international_number: international_number
|
|
||||||
});
|
|
||||||
tokens = tokens.concat(this.id, national_number, international_number);
|
|
||||||
} catch(ex) {
|
|
||||||
return ex;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
this.set({tokens: tokens});
|
this.set({tokens: tokens});
|
||||||
},
|
},
|
||||||
|
|
|
@ -22,20 +22,18 @@
|
||||||
|
|
||||||
validateNumber: function() {
|
validateNumber: function() {
|
||||||
var input = this.$('input.number');
|
var input = this.$('input.number');
|
||||||
try {
|
var regionCode = this.$('li.active').attr('data-country-code').toUpperCase();
|
||||||
var regionCode = this.$('li.active').attr('data-country-code').toUpperCase();
|
var number = input.val();
|
||||||
var number = input.val();
|
|
||||||
|
|
||||||
var parsedNumber = libphonenumber.util.verifyNumber(number, regionCode);
|
|
||||||
|
|
||||||
|
var parsedNumber = libphonenumber.util.parseNumber(number, regionCode);
|
||||||
|
if (parsedNumber.isValidNumber) {
|
||||||
this.$('.number-container').removeClass('invalid');
|
this.$('.number-container').removeClass('invalid');
|
||||||
this.$('.number-container').addClass('valid');
|
this.$('.number-container').addClass('valid');
|
||||||
return parsedNumber;
|
return parsedNumber.e164;
|
||||||
} catch(e) {
|
} else {
|
||||||
this.$('.number-container').removeClass('valid');
|
this.$('.number-container').removeClass('valid');
|
||||||
} finally {
|
|
||||||
input.trigger('validation');
|
|
||||||
}
|
}
|
||||||
|
input.trigger('validation');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
|
@ -127,6 +127,7 @@
|
||||||
<script type="text/javascript" src="../js/views/new_conversation_view.js" data-cover></script>
|
<script type="text/javascript" src="../js/views/new_conversation_view.js" data-cover></script>
|
||||||
<script type="text/javascript" src="../js/views/conversation_search_view.js"></script>
|
<script type="text/javascript" src="../js/views/conversation_search_view.js"></script>
|
||||||
|
|
||||||
|
<script type="text/javascript" src="libphonenumber_util_test.js"></script>
|
||||||
<script type="text/javascript" src="views/whisper_view_test.js"></script>
|
<script type="text/javascript" src="views/whisper_view_test.js"></script>
|
||||||
<script type="text/javascript" src="views/group_update_view_test.js"></script>
|
<script type="text/javascript" src="views/group_update_view_test.js"></script>
|
||||||
<script type="text/javascript" src="views/message_view_test.js"></script>
|
<script type="text/javascript" src="views/message_view_test.js"></script>
|
||||||
|
|
29
test/libphonenumber_util_test.js
Normal file
29
test/libphonenumber_util_test.js
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* vim: ts=4:sw=4:expandtab
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
'use strict';
|
||||||
|
describe('libphonenumber util', function() {
|
||||||
|
describe('parseNumber', function() {
|
||||||
|
it('numbers with + are valid without providing regionCode', function() {
|
||||||
|
var result = libphonenumber.util.parseNumber('+14155555555');
|
||||||
|
assert.isTrue(result.isValidNumber);
|
||||||
|
assert.strictEqual(result.nationalNumber, '4155555555');
|
||||||
|
assert.strictEqual(result.e164, '+14155555555');
|
||||||
|
assert.strictEqual(result.regionCode, 'US');
|
||||||
|
assert.strictEqual(result.countryCode, '1');
|
||||||
|
});
|
||||||
|
it('variant numbers with the right regionCode are valid', function() {
|
||||||
|
[ '4155555555', '14155555555', '+14155555555', ].forEach(function(number) {
|
||||||
|
var result = libphonenumber.util.parseNumber(number, 'US');
|
||||||
|
assert.isTrue(result.isValidNumber);
|
||||||
|
assert.strictEqual(result.nationalNumber, '4155555555');
|
||||||
|
assert.strictEqual(result.e164, '+14155555555');
|
||||||
|
assert.strictEqual(result.regionCode, 'US');
|
||||||
|
assert.strictEqual(result.countryCode, '1');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})();
|
|
@ -153,6 +153,43 @@
|
||||||
convo.revokeAvatarUrl();
|
convo.revokeAvatarUrl();
|
||||||
assert.notOk(convo.avatarUrl);
|
assert.notOk(convo.avatarUrl);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('phone number parsing', function() {
|
||||||
|
after(function() { storage.remove('regionCode'); });
|
||||||
|
function checkAttributes(number) {
|
||||||
|
var convo = new Whisper.ConversationCollection().add({type: 'private'});
|
||||||
|
convo.set('id', number);
|
||||||
|
convo.validate(convo.attributes);
|
||||||
|
assert.strictEqual(convo.get('id'), '+14155555555', number);
|
||||||
|
}
|
||||||
|
it('processes the phone number when validating', function() {
|
||||||
|
[ '+14155555555', ].forEach(checkAttributes);
|
||||||
|
});
|
||||||
|
it('defaults to the local regionCode', function() {
|
||||||
|
storage.put('regionCode', 'US');
|
||||||
|
[ '14155555555', '4155555555' ].forEach(checkAttributes);
|
||||||
|
});
|
||||||
|
it('works with common phone number formats', function() {
|
||||||
|
storage.put('regionCode', 'US');
|
||||||
|
[
|
||||||
|
'415 555 5555',
|
||||||
|
'415-555-5555',
|
||||||
|
'(415) 555 5555',
|
||||||
|
'(415) 555-5555',
|
||||||
|
'1 415 555 5555',
|
||||||
|
'1 415-555-5555',
|
||||||
|
'1 (415) 555 5555',
|
||||||
|
'1 (415) 555-5555',
|
||||||
|
'+1 415 555 5555',
|
||||||
|
'+1 415-555-5555',
|
||||||
|
'+1 (415) 555 5555',
|
||||||
|
'+1 (415) 555-5555',
|
||||||
|
|
||||||
|
].forEach(checkAttributes);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
})();;
|
})();;
|
||||||
|
|
Loading…
Reference in a new issue