diff --git a/.eslintrc b/.eslintrc new file mode 100644 index 0000000..6cfc870 --- /dev/null +++ b/.eslintrc @@ -0,0 +1,177 @@ +/** + * 0 - turn the rule off + * 1 - turn the rule on as a warning (doesn't affect exit code) + * 2 - turn the rule on as an error (exit code will be 1) + * + * Meteor Style Guide: https://github.com/meteor/meteor/wiki/Meteor-Style-Guide + * + */ + +{ + "parser": "babel-eslint", + "env": { + "browser": true, + "node": true + }, + "ecmaFeatures": { + "arrowFunctions": true, + "blockBindings": true, + "classes": true, + "defaultParams": true, + "destructuring": true, + "forOf": true, + "generators": false, + "modules": true, + "objectLiteralComputedProperties": true, + "objectLiteralDuplicateProperties": false, + "objectLiteralShorthandMethods": true, + "objectLiteralShorthandProperties": true, + "spread": true, + "superInFunctions": true, + "templateStrings": true, + "jsx": true + }, + "rules": { + /** + * Strict mode + */ + // babel inserts "use strict"; for us + // http://eslint.org/docs/rules/strict + "strict": 0, + + /** + * ES6 + */ + "no-var": 1, // http://eslint.org/docs/rules/no-var + + /** + * Variables + */ + "no-shadow": 2, // http://eslint.org/docs/rules/no-shadow + "no-shadow-restricted-names": 2, // http://eslint.org/docs/rules/no-shadow-restricted-names + "no-unused-vars": [2, { // http://eslint.org/docs/rules/no-unused-vars + "vars": "local", + "args": "after-used" + }], + "no-use-before-define": [2, "nofunc"], // http://eslint.org/docs/rules/no-use-before-define + + /** + * Possible errors + */ + "comma-dangle": [1, "never"], // http://eslint.org/docs/rules/comma-dangle + "no-cond-assign": [2, "always"], // http://eslint.org/docs/rules/no-cond-assign + "no-console": 1, // http://eslint.org/docs/rules/no-console + "no-debugger": 1, // http://eslint.org/docs/rules/no-debugger + "no-alert": 1, // http://eslint.org/docs/rules/no-alert + "no-constant-condition": 1, // http://eslint.org/docs/rules/no-constant-condition + "no-dupe-keys": 2, // http://eslint.org/docs/rules/no-dupe-keys + "no-duplicate-case": 2, // http://eslint.org/docs/rules/no-duplicate-case + "no-empty": 2, // http://eslint.org/docs/rules/no-empty + "no-ex-assign": 2, // http://eslint.org/docs/rules/no-ex-assign + "no-extra-boolean-cast": 0, // http://eslint.org/docs/rules/no-extra-boolean-cast + "no-extra-semi": 2, // http://eslint.org/docs/rules/no-extra-semi + "no-func-assign": 2, // http://eslint.org/docs/rules/no-func-assign + "no-inner-declarations": 2, // http://eslint.org/docs/rules/no-inner-declarations + "no-invalid-regexp": 2, // http://eslint.org/docs/rules/no-invalid-regexp + "no-irregular-whitespace": 2, // http://eslint.org/docs/rules/no-irregular-whitespace + "no-obj-calls": 2, // http://eslint.org/docs/rules/no-obj-calls + "quote-props": [2, "as-needed", { "keywords": true, "unnecessary": false }], // http://eslint.org/docs/rules/quote-props (previously known as no-reserved-keys) + "no-sparse-arrays": 2, // http://eslint.org/docs/rules/no-sparse-arrays + "no-unreachable": 2, // http://eslint.org/docs/rules/no-unreachable + "use-isnan": 2, // http://eslint.org/docs/rules/use-isnan + "block-scoped-var": 0, // http://eslint.org/docs/rules/block-scoped-var + + /** + * Best practices + */ + "consistent-return": 2, // http://eslint.org/docs/rules/consistent-return + "curly": [2, "multi-line"], // http://eslint.org/docs/rules/curly + "default-case": 2, // http://eslint.org/docs/rules/default-case + "dot-notation": [2, { // http://eslint.org/docs/rules/dot-notation + "allowKeywords": true + }], + "eqeqeq": 2, // http://eslint.org/docs/rules/eqeqeq + "guard-for-in": 2, // http://eslint.org/docs/rules/guard-for-in + "no-caller": 2, // http://eslint.org/docs/rules/no-caller + //"no-else-return": 2, // http://eslint.org/docs/rules/no-else-return + "no-eq-null": 2, // http://eslint.org/docs/rules/no-eq-null + "no-eval": 2, // http://eslint.org/docs/rules/no-eval + "no-extend-native": 2, // http://eslint.org/docs/rules/no-extend-native + "no-extra-bind": 2, // http://eslint.org/docs/rules/no-extra-bind + "no-fallthrough": 2, // http://eslint.org/docs/rules/no-fallthrough + "no-floating-decimal": 2, // http://eslint.org/docs/rules/no-floating-decimal + "no-implied-eval": 2, // http://eslint.org/docs/rules/no-implied-eval + "no-lone-blocks": 2, // http://eslint.org/docs/rules/no-lone-blocks + "no-loop-func": 2, // http://eslint.org/docs/rules/no-loop-func + "no-multi-str": 2, // http://eslint.org/docs/rules/no-multi-str + "no-native-reassign": 2, // http://eslint.org/docs/rules/no-native-reassign + "no-new": 2, // http://eslint.org/docs/rules/no-new + "no-new-func": 2, // http://eslint.org/docs/rules/no-new-func + "no-new-wrappers": 2, // http://eslint.org/docs/rules/no-new-wrappers + "no-octal": 2, // http://eslint.org/docs/rules/no-octal + "no-octal-escape": 2, // http://eslint.org/docs/rules/no-octal-escape + "no-param-reassign": 2, // http://eslint.org/docs/rules/no-param-reassign + "no-proto": 2, // http://eslint.org/docs/rules/no-proto + "no-redeclare": 2, // http://eslint.org/docs/rules/no-redeclare + "no-return-assign": 2, // http://eslint.org/docs/rules/no-return-assign + "no-script-url": 2, // http://eslint.org/docs/rules/no-script-url + "no-self-compare": 2, // http://eslint.org/docs/rules/no-self-compare + "no-sequences": 2, // http://eslint.org/docs/rules/no-sequences + "no-throw-literal": 2, // http://eslint.org/docs/rules/no-throw-literal + "no-with": 2, // http://eslint.org/docs/rules/no-with + "radix": 2, // http://eslint.org/docs/rules/radix + "vars-on-top": 1, // http://eslint.org/docs/rules/vars-on-top + "wrap-iife": [2, "any"], // http://eslint.org/docs/rules/wrap-iife + "yoda": 2, // http://eslint.org/docs/rules/yoda + "max-len": [1, 200, 2], // http://eslint.org/docs/rules/max-len + + /** + * Style + */ + "indent": [2, 2, {"VariableDeclarator": 2}], // http://eslint.org/docs/rules/indent + "brace-style": [2, // http://eslint.org/docs/rules/brace-style + "1tbs", { + "allowSingleLine": true + }], + "camelcase": [2, { // http://eslint.org/docs/rules/camelcase + "properties": "never" + }], + "comma-spacing": [2, { // http://eslint.org/docs/rules/comma-spacing + "before": false, + "after": true + }], + "comma-style": [2, "last"], // http://eslint.org/docs/rules/comma-style + "eol-last": 2, // http://eslint.org/docs/rules/eol-last + "func-names": 0, // http://eslint.org/docs/rules/func-names + "func-style": [2, "expression"], // http://eslint.org/docs/rules/func-style + "key-spacing": [2, { // http://eslint.org/docs/rules/key-spacing + "beforeColon": false, + "afterColon": true + }], + "new-cap": [2, { // http://eslint.org/docs/rules/new-cap + "newIsCap": true + }], + "no-multiple-empty-lines": [2, { // http://eslint.org/docs/rules/no-multiple-empty-lines + "max": 2 + }], + "no-nested-ternary": 2, // http://eslint.org/docs/rules/no-nested-ternary + "no-new-object": 2, // http://eslint.org/docs/rules/no-new-object + "no-array-constructor": 2, // http://eslint.org/docs/rules/no-array-constructor + "no-spaced-func": 2, // http://eslint.org/docs/rules/no-spaced-func + "no-trailing-spaces": 2, // http://eslint.org/docs/rules/no-trailing-spaces + "no-extra-parens": 0, // http://eslint.org/docs/rules/no-extra-parens (previously known as no-wrap-func) + "no-underscore-dangle": 0, // http://eslint.org/docs/rules/no-underscore-dangle + "one-var": [1, "never"], // http://eslint.org/docs/rules/one-var + "semi": [2, "always"], // http://eslint.org/docs/rules/semi + "semi-spacing": [2, { // http://eslint.org/docs/rules/semi-spacing + "before": false, + "after": true + }], + "space-after-keywords": 2, // http://eslint.org/docs/rules/space-after-keywords + "space-before-blocks": 2, // http://eslint.org/docs/rules/space-before-blocks + "space-before-function-paren": [2, "never"], // http://eslint.org/docs/rules/space-before-function-paren + "space-infix-ops": 2, // http://eslint.org/docs/rules/space-infix-ops + "space-return-throw-case": 2, // http://eslint.org/docs/rules/space-return-throw-case + "spaced-comment": 2, // http://eslint.org/docs/rules/spaced-comment (previously known as spaced-line-comment) + } +} diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..cedea3b --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +.build* +local-packages.json +packages +.idea \ No newline at end of file diff --git a/.versions b/.versions new file mode 100644 index 0000000..1ca33bc --- /dev/null +++ b/.versions @@ -0,0 +1,60 @@ +babel-compiler@5.8.24_1 +babel-runtime@0.1.4 +base64@1.0.4 +binary-heap@1.0.4 +blaze@2.1.3 +blaze-tools@1.0.4 +boilerplate-generator@1.0.4 +caching-compiler@1.0.0 +callback-hook@1.0.4 +check@1.1.0 +coffeescript@1.0.11 +ddp@1.2.2 +ddp-client@1.2.1 +ddp-common@1.2.2 +ddp-server@1.2.2 +deps@1.0.9 +diff-sequence@1.0.1 +ecmascript@0.1.6 +ecmascript-runtime@0.2.6 +ejson@1.0.7 +fongandrew:find-and-modify@0.2.1 +geojson-utils@1.0.4 +html-tools@1.0.5 +htmljs@1.0.5 +id-map@1.0.4 +jquery@1.11.4 +local-test:space:vo-i18n@0.1.0 +logging@1.0.8 +meteor@1.1.10 +minimongo@1.0.10 +mongo@1.1.3 +mongo-id@1.0.1 +npm-mongo@1.4.39_1 +observe-sequence@1.0.7 +ordered-dict@1.0.4 +practicalmeteor:chai@2.1.0_1 +practicalmeteor:loglevel@1.2.0_2 +practicalmeteor:munit@2.1.5 +practicalmeteor:sinon@1.14.1_2 +promise@0.5.1 +random@1.0.5 +reactive-dict@1.1.3 +reactive-var@1.0.6 +retry@1.0.4 +routepolicy@1.0.6 +space:base@4.0.0 +space:domain@0.1.0 +space:messaging@3.0.1 +space:testing@3.0.1 +space:testing-messaging@3.0.0 +space:vo-i18n@0.1.0 +spacebars@1.0.7 +spacebars-compiler@1.0.7 +test-helpers@1.0.5 +tinytest@1.0.6 +tracker@1.0.9 +ui@1.0.8 +underscore@1.0.4 +webapp@1.2.3 +webapp-hashing@1.0.5 diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..8c2c946 --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,5 @@ +Changelog +========= + +### 0.1.0 +First release of the Javascript port of `Language` and `Country` value objects. diff --git a/README.md b/README.md new file mode 100644 index 0000000..e3abae9 --- /dev/null +++ b/README.md @@ -0,0 +1,24 @@ +# Value Objects for i18n domains + +[![Build Status](https://travis-ci.org/meteor-space/vo-i18n.svg?branch=master)](https://travis-ci.org/meteor-space/vo-i18n) +[![Join the chat at https://gitter.im/meteor-space/general](https://badges.gitter.im/Join%20Chat.svg)](https://gitter.im/meteor-space/general?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) + +## Installation +`meteor add space:vo-i18n` + +## Documentation +Please look through the tests to get a feeling what this package can do for you. +We hope to find time to write some more documentation together soon ;-) + +## Contributing +In lieu of a formal styleguide, take care to maintain the existing coding style. +Add unit / integration tests for any new or changed functionality. + +## Run the tests +`meteor test-packages ./` + +## Release History +You can find the release history in the [changelog](https://github.com/meteor-space/vo-i18n/blob/master/CHANGELOG.md) + +## License +Licensed under the MIT license. diff --git a/ci.sh b/ci.sh new file mode 100755 index 0000000..bf44aff --- /dev/null +++ b/ci.sh @@ -0,0 +1,9 @@ +#!/usr/bin/env bash + +export PACKAGE_DIRS='packages' + +if [ "$PORT" ]; then + spacejam test-packages ./ --port $PORT +else + spacejam test-packages ./ +fi \ No newline at end of file diff --git a/circle.yml b/circle.yml new file mode 100644 index 0000000..7eb2a03 --- /dev/null +++ b/circle.yml @@ -0,0 +1,16 @@ +machine: + node: + version: 0.10.33 + environment: + PACKAGE_DIRS: packages + pre: + - curl https://install.meteor.com | /bin/sh +dependencies: + pre: + - npm install -g mgp + - npm install -g spacejam + override: + - mgp +test: + override: + - ./ci.sh diff --git a/git-packages.json b/git-packages.json new file mode 100644 index 0000000..232cdcc --- /dev/null +++ b/git-packages.json @@ -0,0 +1,22 @@ +{ + "space:base": { + "git":"https://github.com/meteor-space/base.git", + "version": "89534cec7275ea0f50c0de88f84bba9216be86c6" + }, + "space:messaging": { + "git":"https://github.com/meteor-space/messaging.git", + "version": "bbaf57c511f7230c662d2ea4473b056148c82787" + }, + "space:domain": { + "git":"https://github.com/meteor-space/domain.git", + "version": "c3915404196c201b3db69effb844b16bc1ceb74a" + }, + "space:testing": { + "git":"https://github.com/meteor-space/testing.git", + "version": "53f24417c325500e1b836b88d5f03a17b2d97585" + }, + "space:testing-messaging": { + "git":"https://github.com/meteor-space/testing-messaging.git", + "version": "81aee53c36b724818f1581572c1206c19d776ae9" + } +} diff --git a/package.js b/package.js new file mode 100644 index 0000000..e3a63fe --- /dev/null +++ b/package.js @@ -0,0 +1,47 @@ + +Package.describe({ + name: 'space:vo-i18n', + summary: 'Value Objects for i18n domains.', + version: '0.1.0', + git: 'https://github.com/meteor-space/vo-i18n.git', + documentation: 'README.md' +}); + +Package.onUse(function(api) { + + api.versionsFrom('1.2.0.1'); + + api.use([ + 'check', + 'ecmascript', + 'space:domain@0.1.0' + ]); + + api.addFiles([ + 'source/language.js', + 'source/country.js' + ]); + + api.export('Language'); + api.export('Country'); + +}); + +Package.onTest(function(api) { + + api.use([ + 'ecmascript', + 'check', + 'ejson', + 'space:vo-i18n', + 'practicalmeteor:munit@2.1.5', + 'space:testing@3.0.1', + 'space:testing-messaging@3.0.0' + ]); + + api.addFiles([ + 'tests/language.unit.js', + 'tests/country.unit.js' + ]); + +}); diff --git a/source/country.js b/source/country.js new file mode 100644 index 0000000..55b14f0 --- /dev/null +++ b/source/country.js @@ -0,0 +1,83 @@ +Country = Space.domain.ValueObject.extend('Country', { + + Constructor(data) { + + // Allow to provide another object with country attribute as param + let country = (data && data.code) ? data.code : data; + + if(!Country.isValidCountryCode(country)) { + throw new Error(Country.ERRORS.invalidCountryCode(country)); + } + + this.code = country; + Object.freeze(this); + }, + + // Defines the EJSON fields that are automatically serialized + fields() { + return { + code: String + }; + }, + + isEuropean() { + return Country.isEuropean(this.code); + }, + + toString() { + return this.code; + } + +}); + +Country.ERRORS = { + invalidCountryCode(code) { + return `Invalid country code '${code}'`; + } +}; + +Country.isEuropean = function(code) { + return Country.EUROPEAN_COUNTRIES.indexOf(code) > -1; +}; + +Country.isValidCountryCode = function(code) { + return Country.ISO_COUNTRY_CODES.indexOf(code) > -1; +}; + +Country.getAll = function() { + return Country.ISO_COUNTRY_CODES.slice(); +}; + +Country.getEuropean = function() { + return Country.EUROPEAN_COUNTRIES.slice(); +}; + +Country.ISO_COUNTRY_CODES = [ + 'AF', 'AX', 'AL', 'DZ', 'AS', 'AD', 'AO', 'AI', 'AQ', 'AG', 'AR', 'AM', + 'AW', 'AU', 'AT', 'AZ', 'BS', 'BH', 'BD', 'BB', 'BY', 'BE', 'BZ', 'BJ', + 'BM', 'BT', 'BO', 'BQ', 'BA', 'BW', 'BV', 'BR', 'IO', 'BN', 'BG', 'BF', + 'BI', 'KH', 'CM', 'CA', 'CV', 'KY', 'CF', 'TD', 'CL', 'CN', 'CX', 'CC', + 'CO', 'KM', 'CG', 'CD', 'CK', 'CR', 'CI', 'HR', 'CU', 'CW', 'CY', 'CZ', + 'DK', 'DJ', 'DM', 'DO', 'EC', 'EG', 'SV', 'GQ', 'ER', 'EE', 'ET', 'FK', + 'FO', 'FJ', 'FI', 'FR', 'GF', 'PF', 'TF', 'GA', 'GM', 'GE', 'DE', 'GH', + 'GI', 'GR', 'GL', 'GD', 'GP', 'GU', 'GT', 'GG', 'GN', 'GW', 'GY', 'HT', + 'HM', 'VA', 'HN', 'HK', 'HU', 'IS', 'IN', 'ID', 'IR', 'IQ', 'IE', 'IM', + 'IL', 'IT', 'JM', 'JP', 'JE', 'JO', 'KZ', 'KE', 'KI', 'KP', 'KR', 'KW', + 'KG', 'LA', 'LV', 'LB', 'LS', 'LR', 'LY', 'LI', 'LT', 'LU', 'MO', 'MK', + 'MG', 'MW', 'MY', 'MV', 'ML', 'MT', 'MH', 'MQ', 'MR', 'MU', 'YT', 'MX', + 'FM', 'MD', 'MC', 'MN', 'ME', 'MS', 'MA', 'MZ', 'MM', 'NA', 'NR', 'NP', + 'NL', 'NC', 'NZ', 'NI', 'NE', 'NG', 'NU', 'NF', 'MP', 'NO', 'OM', 'PK', + 'PW', 'PS', 'PA', 'PG', 'PY', 'PE', 'PH', 'PN', 'PL', 'PT', 'PR', 'QA', + 'RE', 'RO', 'RU', 'RW', 'BL', 'SH', 'KN', 'LC', 'MF', 'PM', 'VC', 'WS', + 'SM', 'ST', 'SA', 'SN', 'RS', 'SC', 'SL', 'SG', 'SX', 'SK', 'SI', 'SB', + 'SO', 'ZA', 'GS', 'SS', 'ES', 'LK', 'SD', 'SR', 'SJ', 'SZ', 'SE', 'CH', + 'SY', 'TW', 'TJ', 'TZ', 'TH', 'TL', 'TG', 'TK', 'TO', 'TT', 'TN', 'TR', + 'TM', 'TC', 'TV', 'UG', 'UA', 'AE', 'GB', 'US', 'UM', 'UY', 'UZ', 'VU', + 'VE', 'VN', 'VG', 'VI', 'WF', 'EH', 'YE', 'ZM', 'ZW' +]; + +Country.EUROPEAN_COUNTRIES = [ + 'AT','BE', 'BG', 'HR', 'CY', 'CZ', 'DK', 'EE', 'FI', 'FR', 'DE', 'GR', + 'HU', 'IE', 'IT', 'LV', 'LT', 'LU', 'MT', 'NL', 'PL', 'PT', 'RO', 'SK', + 'SI', 'ES', 'SE', 'GB' +]; diff --git a/source/language.js b/source/language.js new file mode 100644 index 0000000..b2b75dc --- /dev/null +++ b/source/language.js @@ -0,0 +1,52 @@ +Language = Space.domain.ValueObject.extend('Language', { + + Constructor(data) { + + // Allow to provide another object with code attribute as param + let language = (data && data.code) ? data.code : data; + + if(!Language.isValid(language)) { + throw new Error(Language.ERRORS.invalidLanguageCode(language)); + } + + this.code = language; + Object.freeze(this); + }, + + // Defines the EJSON fields that are automatically serialized + fields() { + return { + code: String + }; + }, + + toString() { + return this.code; + } + +}); + +Language.ERRORS = { + invalidLanguageCode(code) { + return `Invalid language code '${code}' given.`; + } +}; + +Language.isValid = function (code) { + return Language.ISO_LANGUAGE_CODES.indexOf(code) > -1; +}; + +Language.ISO_LANGUAGE_CODES = [ + 'om', 'ab', 'aa', 'af', 'sq', 'am', 'ar', 'hy', 'as', 'ay', 'az', 'ba', + 'eu', 'bn', 'dz', 'bh', 'bi', 'br', 'bg', 'my', 'be', 'km', 'ca', 'zh', + 'co', 'hr', 'cs', 'da', 'nl', 'en', 'eo', 'et', 'fo', 'fj', 'fi', 'fr', + 'fy', 'gl', 'ka', 'de', 'el', 'kl', 'gn', 'gu', 'ha', 'he', 'hi', 'hu', + 'is', 'id', 'ia', 'ie', 'ik', 'iu', 'ga', 'it', 'ja', 'jw', 'kn', 'ks', + 'kk', 'rw', 'ky', 'rn', 'ko', 'ku', 'lo', 'la', 'lv', 'ln', 'lt', 'mk', + 'mg', 'ms', 'ml', 'mt', 'mi', 'mr', 'mo', 'mn', 'na', 'ne', 'no', 'oc', + 'or', 'ps', 'fa', 'pl', 'pt', 'pa', 'qu', 'rm', 'ro', 'ru', 'sm', 'sg', + 'sa', 'Ga', 'gd', 'sr', 'sh', 'st', 'tn', 'sn', 'sd', 'si', 'ss', 'sk', + 'sl', 'so', 'es', 'su', 'sw', 'sv', 'tl', 'tg', 'ta', 'tt', 'te', 'th', + 'bo', 'ti', 'to', 'ts', 'tr', 'tk', 'tw', 'ug', 'uk', 'ur', 'uz', 'vi', + 'vo', 'cy', 'wo', 'xh', 'yi', 'yo', 'za', 'zu' +]; diff --git a/test.sh b/test.sh new file mode 100755 index 0000000..7b9cd76 --- /dev/null +++ b/test.sh @@ -0,0 +1,9 @@ +#!/usr/bin/env bash + +export PACKAGE_DIRS='packages' + +if [ "$PORT" ]; then + meteor test-packages ./ --port $PORT +else + meteor test-packages ./ +fi diff --git a/tests/country.unit.js b/tests/country.unit.js new file mode 100644 index 0000000..9c42292 --- /dev/null +++ b/tests/country.unit.js @@ -0,0 +1,63 @@ +describe("Country", function() { + + beforeEach(function() { + this.austria = new Country('AT'); + }); + + it('is serializable', function() { + var copy = EJSON.parse(EJSON.stringify(this.austria)); + expect(copy.equals(this.austria)).to.be.true; + }); + + describe('construction', function() { + + it('takes a country code and assigns it', function() { + expect(this.austria.code).to.equal('AT'); + }); + + it('throws error if invalid country code is assigned', function() { + expect(function() { + new Country('XX'); + }).to["throw"]("Invalid country code 'XX'"); + }); + + }); + + describe('country code validation', function() { + + it('returns true for any valid country code', function() { + for (var i = 0, len = Country.ISO_COUNTRY_CODES.length; i < len; i++) { + var code = Country.ISO_COUNTRY_CODES[i]; + expect(Country.isValidCountryCode(code)).to.be.true; + } + }); + + }); + + describe('checking for european countries', function() { + + it('returns true for any European country code', function() { + for (var i = 0, len = Country.EUROPEAN_COUNTRIES.length; i < len; i++) { + var code = Country.EUROPEAN_COUNTRIES[i]; + expect(Country.isEuropean(code)).to.be.true; + } + }); + + it('returns true if it is an european country', function() { + for (var i = 0, len = Country.EUROPEAN_COUNTRIES.length; i < len; i++) { + var code = Country.EUROPEAN_COUNTRIES[i]; + var country = new Country(code); + expect(country.isEuropean()).to.be.true; + } + }); + + }); + + describe('immutability', function() { + + it('freezes itself', function() { + expect(Object.isFrozen(this.austria)).to.be.true; + }); + + }); +}); diff --git a/tests/language.unit.js b/tests/language.unit.js new file mode 100644 index 0000000..f3beda1 --- /dev/null +++ b/tests/language.unit.js @@ -0,0 +1,34 @@ +describe("Language", function() { + + beforeEach(function() { + this.code = 'de'; + this.language = new Language(this.code); + }); + + it('is serializable', function() { + var copy = EJSON.parse(EJSON.stringify(this.language)); + expect(copy.equals(this.language)).to.be.true; + }); + + describe('construction', function() { + + it('takes a language code and assigns it', function() { + expect(this.language.code).to.equal(this.code); + }); + + it('it only takes valid ISO language codes', function() { + expect(function() { + new Language('XX'); + }).to["throw"]("Invalid language code 'XX' given."); + }); + + }); + + describe('immutability', function() { + + it('freezes itself', function() { + expect(Object.isFrozen(this.language)).to.be.true; + }); + + }); +});