mirror of
				https://github.com/HabitRPG/habitica.git
				synced 2025-10-26 10:42:52 +01:00 
			
		
		
		
	Compare commits
	
		
			469 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f31a82c8f2 | ||
|  | 8bc02e82ee | ||
|  | 9040f9f04e | ||
|  | ff82c37d5f | ||
|  | 37364b0700 | ||
|  | 11cfb3920a | ||
|  | f5468d3771 | ||
|  | 99882d09ab | ||
|  | 7034d135d5 | ||
|  | 034c0c9bb5 | ||
|  | e0711655f0 | ||
|  | 71e162eed5 | ||
|  | 8eac8732c5 | ||
|  | 896a1b74b6 | ||
|  | 3b36046a6a | ||
|  | 07991817e7 | ||
|  | 47b75156fa | ||
|  | c630486fef | ||
|  | 0a070316b5 | ||
|  | f6b34e85df | ||
|  | 2946f0df15 | ||
|  | 614d9a920a | ||
|  | 454524fb5b | ||
|  | abc0777412 | ||
|  | 6972eb8f8f | ||
|  | 535ee2b2a7 | ||
|  | c9755bee7c | ||
|  | f810fff6fc | ||
|  | 5bbe59c52d | ||
|  | 3f52401384 | ||
|  | e7944b3d98 | ||
|  | 08e925e3da | ||
|  | 16c9e42ad8 | ||
|  | 0e1d00c95f | ||
|  | 166f4683ca | ||
|  | 1fcc0d8d3a | ||
|  | 8a4c4e10f1 | ||
|  | 18ed0fe446 | ||
|  | 1f9ebeb629 | ||
|  | b9d83122d1 | ||
|  | d1f7e64156 | ||
|  | 0f8e7416f8 | ||
|  | 1c8b0f92df | ||
|  | 75e5b20f93 | ||
|  | f9db432794 | ||
|  | 6cec7cbba2 | ||
|  | f76d097313 | ||
|  | 59af471438 | ||
|  | d0da303b7d | ||
|  | 596383e7a8 | ||
|  | accba7fc13 | ||
|  | dfc54f1600 | ||
|  | b76ab58e3e | ||
|  | 6f093a94c4 | ||
|  | a9a9e7a4ab | ||
|  | b2058ec23d | ||
|  | 2461f53cf5 | ||
|  | 73fc288f3b | ||
|  | ea78b6feb9 | ||
|  | 7c141614ed | ||
|  | c072935e80 | ||
|  | 54e49ca3b9 | ||
|  | b16a245d61 | ||
|  | 0a8109e496 | ||
|  | cdfcc6419f | ||
|  | 9c25c2452f | ||
|  | 573f2e4732 | ||
|  | 81ffcf9c1b | ||
|  | d8925a8811 | ||
|  | 217c16988b | ||
|  | de7f953b67 | ||
|  | 8ee2a02e73 | ||
|  | d9b573b430 | ||
|  | cbcf5a03e1 | ||
|  | 487523f64b | ||
|  | 74cfc2cf52 | ||
|  | 2d489e870f | ||
|  | 6659d4fa52 | ||
|  | 5471af74fa | ||
|  | 6eb484605a | ||
|  | 8969b755a6 | ||
|  | 0062e5b1f1 | ||
|  | 50b98d8d92 | ||
|  | 7ddf4b1f7b | ||
|  | c91da86b89 | ||
|  | d549fea4ed | ||
|  | a362914f93 | ||
|  | 61001d0e9a | ||
|  | fb95d001ab | ||
|  | bd5c4a08e2 | ||
|  | 34fb90455c | ||
|  | d038d9f9bb | ||
|  | 420d7df4f5 | ||
|  | 3ee8072a6c | ||
|  | 50ebdd1ece | ||
|  | 4a80dcae2e | ||
|  | bc03c1d18a | ||
|  | e5d834b40a | ||
|  | 7f847d322f | ||
|  | ed27ac15c8 | ||
|  | 88188e56d9 | ||
|  | 7170cf05d0 | ||
|  | 22a8d5e94c | ||
|  | f37e5cde57 | ||
|  | 592cfef6c6 | ||
|  | c1bd7f5dc5 | ||
|  | 8437b916c4 | ||
|  | 52e53aa466 | ||
|  | 4471186e09 | ||
|  | 6a2a844e04 | ||
|  | 122d147f07 | ||
|  | 912f00a652 | ||
|  | 627d4330c8 | ||
|  | c7f6794dda | ||
|  | 00cb50a781 | ||
|  | 8be9964483 | ||
|  | 7ea6c911cb | ||
|  | 97a069642d | ||
|  | 26e9827d39 | ||
|  | 88c625fe80 | ||
|  | 4044432fad | ||
|  | 6a767ed70b | ||
|  | 2c2ca4a9c8 | ||
|  | 380ad8c9e5 | ||
|  | f53400f950 | ||
|  | 53c719acbd | ||
|  | 948a5d80c8 | ||
|  | e1f141ee91 | ||
|  | 7a5a278dbb | ||
|  | 08ab4d5900 | ||
|  | b8d5844d0f | ||
|  | 39b81aa685 | ||
|  | 44f196080c | ||
|  | 65bfd74c93 | ||
|  | 5e60a05cac | ||
|  | 59af4a2d3b | ||
|  | 8d273fac5e | ||
|  | b58032d5fb | ||
|  | db4123610f | ||
|  | e4c1d96b59 | ||
|  | fe1f0bf087 | ||
|  | ccd0bec28c | ||
|  | eebf38b5ae | ||
|  | 30cd738635 | ||
|  | 920b07ff12 | ||
|  | a7db15d768 | ||
|  | 93768e70c5 | ||
|  | 3e29b958e3 | ||
|  | ce1bcdeee0 | ||
|  | 971145a72a | ||
|  | 36595e0138 | ||
|  | a1de566c34 | ||
|  | 2b8415fad0 | ||
|  | 2afbc23f6f | ||
|  | a35c1954af | ||
|  | 47c488967c | ||
|  | ee4a05d7ec | ||
|  | 308cd49e9c | ||
|  | 48e51a03d4 | ||
|  | 96f7a192d7 | ||
|  | 1e786412ba | ||
|  | 1739b83609 | ||
|  | 3606b58a1d | ||
|  | 202db599ae | ||
|  | 3aca0343e8 | ||
|  | 97b99c0550 | ||
|  | 0e63f68ed6 | ||
|  | fa142e929f | ||
|  | c4867f1e8e | ||
|  | 5f58fe66de | ||
|  | a0e2d6a05e | ||
|  | b67522e92b | ||
|  | 0e3496395c | ||
|  | 6e7b9f1f93 | ||
|  | e6cf7564b8 | ||
|  | bf424573a4 | ||
|  | ac90a40be5 | ||
|  | 821f84dbe8 | ||
|  | 8fb67e7944 | ||
|  | e81e458e9b | ||
|  | aec23d32f3 | ||
|  | 4f2d066d66 | ||
|  | eaf0c62e16 | ||
|  | fc62db147f | ||
|  | 6c9ff3e8ed | ||
|  | 6ef45a7fd2 | ||
|  | 557212b549 | ||
|  | f8bd116e54 | ||
|  | 9194e8226d | ||
|  | e0140f67be | ||
|  | 1e2fc14db9 | ||
|  | 30082a3929 | ||
|  | 42d7744d12 | ||
|  | 2cbc41d02f | ||
|  | 01ce7712e3 | ||
|  | c52e4a07d4 | ||
|  | 5212ac6394 | ||
|  | 7b5d6b508d | ||
|  | c5a497ef91 | ||
|  | 54bee67e03 | ||
|  | 86ec68bedb | ||
|  | 8223563e76 | ||
|  | 37ab257f5b | ||
|  | 04d7ff13de | ||
|  | 25d07ac0ce | ||
|  | 724e1240a3 | ||
|  | 026e1a5bca | ||
|  | 6443918440 | ||
|  | ac973ee753 | ||
|  | c39b9dc320 | ||
|  | 2132a3a242 | ||
|  | ba52a90d93 | ||
|  | daa4994382 | ||
|  | 12034161b7 | ||
|  | 8438cf0578 | ||
|  | 614848d60b | ||
|  | 79087b27d3 | ||
|  | 5167f847d0 | ||
|  | d3a0348ac7 | ||
|  | de9883c3ac | ||
|  | 3d39718048 | ||
|  | a0c51ee4ca | ||
|  | b2edd1d932 | ||
|  | 6b5f46c5e1 | ||
|  | ad191c2c5c | ||
|  | 4a55d36831 | ||
|  | 959adb05cf | ||
|  | d114b858fd | ||
|  | ae9db7aee3 | ||
|  | 10567d81e2 | ||
|  | ba799c67f9 | ||
|  | 37b890f282 | ||
|  | 196e5f5b95 | ||
|  | 6db412f7e6 | ||
|  | fa60c9a232 | ||
|  | 2c3d268a63 | ||
|  | d4a80a8561 | ||
|  | 388492e1e7 | ||
|  | 8cd695c397 | ||
|  | 355f0fedfb | ||
|  | 38d78de4b3 | ||
|  | 6c64a1cd8c | ||
|  | 128ec5a1b1 | ||
|  | d4d668f640 | ||
|  | 41ccd58f8e | ||
|  | a33299a341 | ||
|  | 9129e22433 | ||
|  | 86d1bdaff1 | ||
|  | 206ed1f155 | ||
|  | eb66e9ec2e | ||
|  | 8db99be017 | ||
|  | c62386e2e5 | ||
|  | 042ac6ac73 | ||
|  | a8655d923a | ||
|  | bbbd1f9f73 | ||
|  | 8fee5a9ba0 | ||
|  | 8df2b1e8c2 | ||
|  | 24cceb1c91 | ||
|  | 21eac3cc94 | ||
|  | e9ce968f88 | ||
|  | 8c283fdbe0 | ||
|  | 69a782a1db | ||
|  | ccaf629228 | ||
|  | 147f2bb28e | ||
|  | 54a4bba228 | ||
|  | 6ee21dcfa9 | ||
|  | 68353fb874 | ||
|  | 68526c07ae | ||
|  | 5f319ca4f6 | ||
|  | 0d84643961 | ||
|  | 8470f16f4f | ||
|  | 1896a8fab0 | ||
|  | 891b5566a9 | ||
|  | c83499545c | ||
|  | 4c837acf88 | ||
|  | 11b223a81e | ||
|  | 17001743e1 | ||
|  | ac451bdb9b | ||
|  | 6af50c9f2f | ||
|  | 5231cb03a8 | ||
|  | f8739b6f37 | ||
|  | e31f62a818 | ||
|  | d5d06c1d2d | ||
|  | 4fa2ef045d | ||
|  | 570a8bf0d5 | ||
|  | b7dfe41e15 | ||
|  | c26696a9eb | ||
|  | f226b5da07 | ||
|  | 63cf5b6be7 | ||
|  | f3a947339c | ||
|  | 1bb8acad5d | ||
|  | 8e04d6e284 | ||
|  | f7415df6ba | ||
|  | f85e1c2dc4 | ||
|  | 33628a0a6a | ||
|  | 5e6541faa6 | ||
|  | c1ed02d383 | ||
|  | 3793e92b80 | ||
|  | e3ce1c5322 | ||
|  | 84b16f28c2 | ||
|  | 9c702505a9 | ||
|  | 27c73e028a | ||
|  | d125b8d2f8 | ||
|  | 451e08ce1c | ||
|  | 16b5b8b8c7 | ||
|  | 30a717148e | ||
|  | bcf9670dbe | ||
|  | 129fccf646 | ||
|  | 9d755c5d5f | ||
|  | 05c43d1f9d | ||
|  | 45df73e4be | ||
|  | eaa00598d0 | ||
|  | 88b14592c5 | ||
|  | 85136675e9 | ||
|  | 4e4181a394 | ||
|  | f93822b0b3 | ||
|  | a864e69042 | ||
|  | 2ccd9eaa1e | ||
|  | 5faf00d489 | ||
|  | f211610f5d | ||
|  | 332f285ea2 | ||
|  | 006159cc9c | ||
|  | 3722452b51 | ||
|  | d6b5d275da | ||
|  | 72073386ec | ||
|  | d34ec62901 | ||
|  | ca73b9af41 | ||
|  | 8b9bf88fa0 | ||
|  | 9133250a42 | ||
|  | e60177f14a | ||
|  | 5f0ef2d8f0 | ||
|  | 770285f10d | ||
|  | 495dd2736c | ||
|  | 4467da980c | ||
|  | 082539b982 | ||
|  | ef7719f91d | ||
|  | f98efd4eb9 | ||
|  | 4a0856c919 | ||
|  | 2adc5c13e4 | ||
|  | cf274310a8 | ||
|  | a2ee73a2e2 | ||
|  | c6c9503e22 | ||
|  | 403ac1ab7e | ||
|  | 63598f497b | ||
|  | 9fcc953b18 | ||
|  | 17408d01a9 | ||
|  | ae786f28a2 | ||
|  | 1effa16b5b | ||
|  | 6b7333927a | ||
|  | 31b439129d | ||
|  | 2de85b937f | ||
|  | 4f963e99dc | ||
|  | 58ce3a9a42 | ||
|  | e45d0c9b80 | ||
|  | 84a20ef4f4 | ||
|  | 59a22805b9 | ||
|  | 95865f5ec8 | ||
|  | 79903d242f | ||
|  | 90959c18cd | ||
|  | 8b2019c292 | ||
|  | 9ab70ca276 | ||
|  | d51aa25470 | ||
|  | 0b2c1e6d2e | ||
|  | 58ee6e9703 | ||
|  | 0044778497 | ||
|  | 46d6590fec | ||
|  | b10f056a73 | ||
|  | eeb890466a | ||
|  | 8d25a5d140 | ||
|  | 3b35a0a203 | ||
|  | d787ad43d3 | ||
|  | 7d7fe6047c | ||
|  | 0ec1a91774 | ||
|  | adf3281bef | ||
|  | ea86b35833 | ||
|  | ade14edcd7 | ||
|  | 3a1888739a | ||
|  | 3b54ce4949 | ||
|  | 4a8aaf7389 | ||
|  | 45eec47b7f | ||
|  | 4b9af8aa86 | ||
|  | 631bbcb786 | ||
|  | 76a10d6cf9 | ||
|  | a1c9ebd661 | ||
|  | 9f06d78db6 | ||
|  | ac98aa9271 | ||
|  | 455f7ac59b | ||
|  | a42cb0e3ab | ||
|  | d05d2fb9d7 | ||
|  | 6c4c5b4697 | ||
|  | 5da87640e4 | ||
|  | fa044ffb44 | ||
|  | 5449652bd2 | ||
|  | c12ae9ea25 | ||
|  | 734a300b92 | ||
|  | 1109ae308d | ||
|  | 8f1d241e83 | ||
|  | acbca4d1dc | ||
|  | 1ea9be8aa2 | ||
|  | ace02893e5 | ||
|  | 1c3e043fac | ||
|  | 71c9e7a685 | ||
|  | fa945c7689 | ||
|  | c54ce96033 | ||
|  | 85c4e93763 | ||
|  | 25e5e78373 | ||
|  | 06181d0a1a | ||
|  | d5a8259fdb | ||
|  | 9db7141853 | ||
|  | ec2a1927a0 | ||
|  | 1c1b0f00ad | ||
|  | fb4d3e44d3 | ||
|  | 37fd062cf9 | ||
|  | 485c3c5c46 | ||
|  | 5007393f24 | ||
|  | e111ac730c | ||
|  | e7c78eabce | ||
|  | 5da7699548 | ||
|  | f42955a0ba | ||
|  | 4d67df4da6 | ||
|  | ab7459f4f3 | ||
|  | 469db7c0e2 | ||
|  | 952e813b30 | ||
|  | f04d05fee1 | ||
|  | 6d9aa43c07 | ||
|  | f527221079 | ||
|  | d9b852e1ea | ||
|  | a1207c1d8d | ||
|  | f4fb90013d | ||
|  | 73a7c0eebc | ||
|  | 1819398f41 | ||
|  | ab14312368 | ||
|  | 690d3e3fd2 | ||
|  | 36f9a4918f | ||
|  | a4b5e27614 | ||
|  | 0abfe86296 | ||
|  | e11c777325 | ||
|  | 63a04f36c9 | ||
|  | e58af6e3ea | ||
|  | 6ba28b5757 | ||
|  | ed607d2bae | ||
|  | 1f7fc594e5 | ||
|  | 45d0a4fac2 | ||
|  | e50bc189aa | ||
|  | 4623bcd877 | ||
|  | 4a368a1128 | ||
|  | bec8cb01e0 | ||
|  | f3c041a561 | ||
|  | c21726ec61 | ||
|  | df69208caa | ||
|  | 08d07cdd67 | ||
|  | a309e48183 | ||
|  | 70c539cc81 | ||
|  | 11f136ac89 | ||
|  | 567d5f74ba | ||
|  | 338781f57b | ||
|  | bd07f3cd38 | ||
|  | 0b735abd44 | ||
|  | a88cdaf1fc | ||
|  | 7cae5f1a37 | ||
|  | e453330535 | ||
|  | b1e5fcdeaf | ||
|  | 10e0848a5c | ||
|  | 64e86bad91 | ||
|  | 21cf5d2321 | ||
|  | a6106a801b | ||
|  | a0803796b2 | ||
|  | 10370ea1dc | ||
|  | 558dd2e4bf | 
							
								
								
									
										4
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| [//]: # (Note: See http://habitica.wikia.com/wiki/Using_Your_Local_Install_to_Modify_Habitica%27s_Website_and_API for more info) | ||||
|  | ||||
| [//]: # (Put Issue # or URL here, if applicable. This will automatically close the issue if your PR is merged in) | ||||
| Fixes put_issue_url_here | ||||
| [//]: # (Put Issue # here, if applicable. This will automatically close the issue if your PR is merged in) | ||||
| Fixes put_#_and_issue_numer_here | ||||
|  | ||||
| ### Changes | ||||
| [//]: # (Describe the changes that were made in detail here. Include pictures if necessary) | ||||
|   | ||||
| @@ -17,3 +17,4 @@ CHANGELOG.md | ||||
| newrelic_agent.log | ||||
| *.swp | ||||
| *.swx | ||||
| website/raw_sprites/** | ||||
|   | ||||
| @@ -20,8 +20,9 @@ env: | ||||
|     - DISABLE_REQUEST_LOGGING=true | ||||
|   matrix: | ||||
|     - TEST="lint" | ||||
|     - TEST="test:api-v3:unit" REQUIRES_SERVER=true COVERAGE=true | ||||
|     - TEST="test:api:unit" REQUIRES_SERVER=true COVERAGE=true | ||||
|     - TEST="test:api-v3:integration" REQUIRES_SERVER=true COVERAGE=true | ||||
|     - TEST="test:api-v4:integration" REQUIRES_SERVER=true COVERAGE=true | ||||
|     - TEST="test:sanity" | ||||
|     - TEST="test:content" COVERAGE=true | ||||
|     - TEST="test:common" COVERAGE=true | ||||
|   | ||||
							
								
								
									
										47
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										47
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,18 +1,29 @@ | ||||
| FROM node:8 | ||||
|  | ||||
| # Install global packages | ||||
| RUN npm install -g gulp-cli mocha | ||||
|  | ||||
| # Clone Habitica repo and install dependencies | ||||
| RUN mkdir -p /usr/src/habitrpg | ||||
| WORKDIR /usr/src/habitrpg | ||||
| RUN git clone https://github.com/HabitRPG/habitica.git /usr/src/habitrpg | ||||
| RUN cp config.json.example config.json | ||||
| RUN npm install | ||||
|  | ||||
| # Create Build dir | ||||
| RUN mkdir -p ./website/build | ||||
|  | ||||
| # Start Habitica | ||||
| EXPOSE 3000 | ||||
| CMD ["npm", "start"] | ||||
| FROM node:8 | ||||
|  | ||||
| ENV ADMIN_EMAIL admin@habitica.com | ||||
| ENV AMAZON_PAYMENTS_CLIENT_ID amzn1.application-oa2-client.68ed9e6904ef438fbc1bf86bf494056e | ||||
| ENV AMAZON_PAYMENTS_SELLER_ID AMQ3SB4SG5E91 | ||||
| ENV AMPLITUDE_KEY e8d4c24b3d6ef3ee73eeba715023dd43 | ||||
| ENV BASE_URL https://habitica.com | ||||
| ENV FACEBOOK_KEY 128307497299777 | ||||
| ENV GA_ID UA-33510635-1 | ||||
| ENV GOOGLE_CLIENT_ID 1035232791481-32vtplgnjnd1aufv3mcu1lthf31795fq.apps.googleusercontent.com | ||||
| ENV NODE_ENV production | ||||
| ENV STRIPE_PUB_KEY pk_85fQ0yMECHNfHTSsZoxZXlPSwSNfA | ||||
|  | ||||
| # Install global packages | ||||
| RUN npm install -g gulp-cli mocha | ||||
|  | ||||
| # Clone Habitica repo and install dependencies | ||||
| RUN mkdir -p /usr/src/habitrpg | ||||
| WORKDIR /usr/src/habitrpg | ||||
| RUN git clone --branch release https://github.com/HabitRPG/habitica.git /usr/src/habitrpg | ||||
| RUN npm install | ||||
| RUN gulp build:prod --force | ||||
|  | ||||
| # Create Build dir | ||||
| RUN mkdir -p ./website/build | ||||
|  | ||||
| # Start Habitica | ||||
| EXPOSE 3000 | ||||
| CMD ["node", "./website/transpiled-babel/index.js"] | ||||
|   | ||||
							
								
								
									
										18
									
								
								Dockerfile-Dev
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								Dockerfile-Dev
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| FROM node:8 | ||||
|  | ||||
| # Install global packages | ||||
| RUN npm install -g gulp-cli mocha | ||||
|  | ||||
| # Clone Habitica repo and install dependencies | ||||
| RUN mkdir -p /usr/src/habitrpg | ||||
| WORKDIR /usr/src/habitrpg | ||||
| RUN git clone https://github.com/HabitRPG/habitica.git /usr/src/habitrpg | ||||
| RUN cp config.json.example config.json | ||||
| RUN npm install | ||||
|  | ||||
| # Create Build dir | ||||
| RUN mkdir -p ./website/build | ||||
|  | ||||
| # Start Habitica | ||||
| EXPOSE 3000 | ||||
| CMD ["npm", "start"] | ||||
| @@ -1,29 +0,0 @@ | ||||
| FROM node:8 | ||||
|  | ||||
| ENV ADMIN_EMAIL admin@habitica.com | ||||
| ENV AMAZON_PAYMENTS_CLIENT_ID amzn1.application-oa2-client.68ed9e6904ef438fbc1bf86bf494056e | ||||
| ENV AMAZON_PAYMENTS_SELLER_ID AMQ3SB4SG5E91 | ||||
| ENV AMPLITUDE_KEY e8d4c24b3d6ef3ee73eeba715023dd43 | ||||
| ENV BASE_URL https://habitica.com | ||||
| ENV FACEBOOK_KEY 128307497299777 | ||||
| ENV GA_ID UA-33510635-1 | ||||
| ENV GOOGLE_CLIENT_ID 1035232791481-32vtplgnjnd1aufv3mcu1lthf31795fq.apps.googleusercontent.com | ||||
| ENV NODE_ENV production | ||||
| ENV STRIPE_PUB_KEY pk_85fQ0yMECHNfHTSsZoxZXlPSwSNfA | ||||
|  | ||||
| # Install global packages | ||||
| RUN npm install -g gulp-cli mocha | ||||
|  | ||||
| # Clone Habitica repo and install dependencies | ||||
| RUN mkdir -p /usr/src/habitrpg | ||||
| WORKDIR /usr/src/habitrpg | ||||
| RUN git clone --branch v4.34.0 https://github.com/HabitRPG/habitica.git /usr/src/habitrpg | ||||
| RUN npm install | ||||
| RUN gulp build:prod --force | ||||
|  | ||||
| # Create Build dir | ||||
| RUN mkdir -p ./website/build | ||||
|  | ||||
| # Start Habitica | ||||
| EXPOSE 3000 | ||||
| CMD ["node", "./website/transpiled-babel/index.js"] | ||||
| @@ -10,4 +10,3 @@ We need more programmers! Your assistance will be greatly appreciated. | ||||
| For an introduction to the technologies used and how the software is organized, refer to [Guidance for Blacksmiths](http://habitica.wikia.com/wiki/Guidance_for_Blacksmiths). | ||||
|  | ||||
| To set up a local install of Habitica for development and testing on various platforms, see [Setting up Habitica Locally](http://habitica.wikia.com/wiki/Setting_up_Habitica_Locally). | ||||
|  | ||||
|   | ||||
| @@ -78,6 +78,9 @@ | ||||
|     "PUSH_CONFIGS": { | ||||
|         "GCM_SERVER_API_KEY": "", | ||||
|         "APN_ENABLED": "false", | ||||
|         "APN_KEY_ID": "xxxxxxxxxx", | ||||
|         "APN_KEY": "xxxxxxxxxx", | ||||
|         "APN_TEAM_ID": "aaabbbcccd", | ||||
|         "FCM_SERVER_API_KEY": "" | ||||
|     }, | ||||
|     "SITE_HTTP_AUTH": { | ||||
| @@ -98,9 +101,9 @@ | ||||
|     }, | ||||
|     "ITUNES_SHARED_SECRET": "aaaabbbbccccddddeeeeffff00001111", | ||||
|     "EMAILS" : { | ||||
|         "COMMUNITY_MANAGER_EMAIL" : "leslie@habitica.com", | ||||
|         "COMMUNITY_MANAGER_EMAIL" : "admin@habitica.com", | ||||
|         "TECH_ASSISTANCE_EMAIL" : "admin@habitica.com", | ||||
|         "PRESS_ENQUIRY_EMAIL" : "leslie@habitica.com" | ||||
|         "PRESS_ENQUIRY_EMAIL" : "admin@habitica.com" | ||||
|     }, | ||||
|     "LOGGLY" : { | ||||
|         "TOKEN" : "example-token", | ||||
| @@ -113,5 +116,5 @@ | ||||
|       "CLOUDKARAFKA_PASSWORD": "", | ||||
|       "CLOUDKARAFKA_TOPIC_PREFIX": "" | ||||
|     }, | ||||
|     "STACK_IMPACT_KEY": "aaaabbbbccccddddeeeeffffgggg111100002222" | ||||
|     "MIGRATION_CONNECT_STRING": "mongodb://localhost:27017/habitrpg?auto_reconnect=true" | ||||
| } | ||||
|   | ||||
							
								
								
									
										100
									
								
								database_reports/20180607_subscriber_histories.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								database_reports/20180607_subscriber_histories.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| import max from 'lodash/max'; | ||||
| import mean from 'lodash/mean'; | ||||
| import monk from 'monk'; | ||||
| import round from 'lodash/round'; | ||||
| import sum from 'lodash/sum'; | ||||
|  | ||||
| /* | ||||
|  * Output data on subscribers' task histories, formatted for CSV. | ||||
|  * User ID,Count of Dailies,Count of Habits,Total History Size,Max History Size,Mean History Size,Median History Size | ||||
|  */ | ||||
| const connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE | ||||
|  | ||||
| let dbUsers = monk(connectionString).get('users', { castIds: false }); | ||||
| let dbTasks = monk(connectionString).get('tasks', { castIds: false }); | ||||
|  | ||||
| function usersReport () { | ||||
|   let allHistoryLengths = []; | ||||
|  | ||||
|   console.info('User ID,Count of Dailies,Count of Habits,Total History Size,Max History Size,Mean History Size,Median History Size'); | ||||
|  | ||||
|   dbUsers.find( | ||||
|     { | ||||
|       $and: | ||||
|         [ | ||||
|           {'purchased.plan.planId': {$ne:null}}, | ||||
|           {'purchased.plan.planId': {$ne:''}}, | ||||
|         ], | ||||
|       $or: | ||||
|         [ | ||||
|           {'purchased.plan.dateTerminated': null}, | ||||
|           {'purchased.plan.dateTerminated': ''}, | ||||
|           {'purchased.plan.dateTerminated': {$gt:new Date()}}, | ||||
|         ], | ||||
|     }, | ||||
|     { | ||||
|       fields: {_id: 1}, | ||||
|     } | ||||
|   ).each((user, {close, pause, resume}) => { | ||||
|     let historyLengths = []; | ||||
|     let habitCount = 0; | ||||
|     let dailyCount = 0; | ||||
|  | ||||
|     pause(); | ||||
|     return dbTasks.find( | ||||
|       { | ||||
|         userId: user._id, | ||||
|         $or: | ||||
|           [ | ||||
|             {type: 'habit'}, | ||||
|             {type: 'daily'}, | ||||
|           ], | ||||
|       }, | ||||
|       { | ||||
|         fields: { | ||||
|           type: 1, | ||||
|           history: 1, | ||||
|         }, | ||||
|       } | ||||
|     ).each((task) => { | ||||
|       if (task.type === 'habit') { | ||||
|         habitCount++; | ||||
|       } | ||||
|       if (task.type === 'daily') { | ||||
|         dailyCount++; | ||||
|       } | ||||
|       if (task.history.length > 0) { | ||||
|         allHistoryLengths.push(task.history.length); | ||||
|         historyLengths.push(task.history.length); | ||||
|       } | ||||
|     }).then(() => { | ||||
|       const totalHistory = sum(historyLengths); | ||||
|       const maxHistory = historyLengths.length > 0 ? max(historyLengths) : 0; | ||||
|       const meanHistory = historyLengths.length > 0 ? round(mean(historyLengths)) : 0; | ||||
|       const medianHistory = historyLengths.length > 0 ? median(historyLengths) : 0; | ||||
|       console.info(`${user._id},${dailyCount},${habitCount},${totalHistory},${maxHistory},${meanHistory},${medianHistory}`); | ||||
|       resume(); | ||||
|     }); | ||||
|   }).then(() => { | ||||
|     console.info(`Total Subscriber History Entries: ${sum(allHistoryLengths)}`); | ||||
|     console.info(`Largest History Size: ${max(allHistoryLengths)}`); | ||||
|     console.info(`Mean History Size: ${round(mean(allHistoryLengths))}`); | ||||
|     console.info(`Median History Size: ${median(allHistoryLengths)}`); | ||||
|     return process.exit(0); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| function median(values) { // https://gist.github.com/caseyjustus/1166258 | ||||
|   values.sort( function(a,b) {return a - b;} ); | ||||
|  | ||||
|   var half = Math.floor(values.length/2); | ||||
|  | ||||
|   if (values.length % 2) { | ||||
|     return values[half]; | ||||
|   } | ||||
|   else { | ||||
|     return (values[half-1] + values[half]) / 2.0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| module.exports = usersReport; | ||||
| @@ -25,7 +25,7 @@ services: | ||||
|       - mongo | ||||
|  | ||||
|   mongo: | ||||
|     image: mongo | ||||
|     image: mongo:3.4 | ||||
|     ports: | ||||
|       - "27017:27017" | ||||
|     networks: | ||||
|   | ||||
| @@ -165,9 +165,9 @@ gulp.task('test:content:safe', gulp.series('test:prepare:build', (cb) => { | ||||
|   pipe(runner); | ||||
| })); | ||||
|  | ||||
| gulp.task('test:api-v3:unit', (done) => { | ||||
| gulp.task('test:api:unit', (done) => { | ||||
|   let runner = exec( | ||||
|     testBin('node_modules/.bin/istanbul cover --dir coverage/api-v3-unit --report lcovonly node_modules/mocha/bin/_mocha -- test/api/v3/unit --recursive --require ./test/helpers/start-server'), | ||||
|     testBin('node_modules/.bin/istanbul cover --dir coverage/api-unit node_modules/mocha/bin/_mocha -- test/api/unit --recursive --require ./test/helpers/start-server'), | ||||
|     (err) => { | ||||
|       if (err) { | ||||
|         process.exit(1); | ||||
| @@ -179,8 +179,8 @@ gulp.task('test:api-v3:unit', (done) => { | ||||
|   pipe(runner); | ||||
| }); | ||||
|  | ||||
| gulp.task('test:api-v3:unit:watch', () => { | ||||
|   return gulp.watch(['website/server/libs/*', 'test/api/v3/unit/**/*', 'website/server/controllers/**/*'], gulp.series('test:api-v3:unit', done => done())); | ||||
| gulp.task('test:api:unit:watch', () => { | ||||
|   return gulp.watch(['website/server/libs/*', 'test/api/v3/unit/**/*', 'website/server/controllers/**/*'], gulp.series('test:api:unit', done => done())); | ||||
| }); | ||||
|  | ||||
| gulp.task('test:api-v3:integration', (done) => { | ||||
| @@ -215,17 +215,43 @@ gulp.task('test:api-v3:integration:separate-server', (done) => { | ||||
|   pipe(runner); | ||||
| }); | ||||
|  | ||||
| gulp.task('test:api-v4:integration', (done) => { | ||||
|   let runner = exec( | ||||
|     testBin('node_modules/.bin/istanbul cover --dir coverage/api-v4-integration --report lcovonly node_modules/mocha/bin/_mocha -- test/api/v4 --recursive --require ./test/helpers/start-server'), | ||||
|     {maxBuffer: 500 * 1024}, | ||||
|     (err) => { | ||||
|       if (err) { | ||||
|         process.exit(1); | ||||
|       } | ||||
|       done(); | ||||
|     } | ||||
|   ); | ||||
|  | ||||
|   pipe(runner); | ||||
| }); | ||||
|  | ||||
| gulp.task('test:api-v4:integration:separate-server', (done) => { | ||||
|   let runner = exec( | ||||
|     testBin('mocha test/api/v4 --recursive --require ./test/helpers/start-server', 'LOAD_SERVER=0'), | ||||
|     {maxBuffer: 500 * 1024}, | ||||
|     (err) => done(err) | ||||
|   ); | ||||
|  | ||||
|   pipe(runner); | ||||
| }); | ||||
|  | ||||
| gulp.task('test', gulp.series( | ||||
|   'test:sanity', | ||||
|   'test:content', | ||||
|   'test:common', | ||||
|   'test:api-v3:unit', | ||||
|   'test:api:unit', | ||||
|   'test:api-v3:integration', | ||||
|   'test:api-v4:integration', | ||||
|   done => done() | ||||
| )); | ||||
|  | ||||
| gulp.task('test:api-v3', gulp.series( | ||||
|   'test:api-v3:unit', | ||||
|   'test:api:unit', | ||||
|   'test:api-v3:integration', | ||||
|   done => done() | ||||
| )); | ||||
| )); | ||||
							
								
								
									
										52
									
								
								migrations/groups/migrate-chat.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								migrations/groups/migrate-chat.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| // @migrationName = 'MigrateGroupChat'; | ||||
| // @authorName = 'TheHollidayInn'; // in case script author needs to know when their ... | ||||
| // @authorUuid = ''; // ... own data is done | ||||
|  | ||||
|  | ||||
| /* | ||||
|  * This migration moves chat off of groups and into their own model | ||||
|  */ | ||||
|  | ||||
| import { model as Group } from '../../website/server/models/group'; | ||||
| import { model as Chat } from '../../website/server/models/chat'; | ||||
|  | ||||
| async function moveGroupChatToModel (skip = 0) { | ||||
|   const groups = await Group.find({}) | ||||
|     .limit(50) | ||||
|     .skip(skip) | ||||
|     .sort({ _id: -1 }) | ||||
|     .exec(); | ||||
|  | ||||
|   if (groups.length === 0) { | ||||
|     console.log('End of groups'); | ||||
|     process.exit(); | ||||
|   } | ||||
|  | ||||
|   const promises = groups.map(group => { | ||||
|     const chatpromises = group.chat.map(message => { | ||||
|       const newChat = new Chat(); | ||||
|       Object.assign(newChat, message); | ||||
|       newChat._id = message.id; | ||||
|       newChat.groupId = group._id; | ||||
|  | ||||
|       return newChat.save(); | ||||
|     }); | ||||
|  | ||||
|     group.chat = []; | ||||
|     chatpromises.push(group.save()); | ||||
|  | ||||
|     return chatpromises; | ||||
|   }); | ||||
|  | ||||
|  | ||||
|   const reducedPromises = promises.reduce((acc, curr) => { | ||||
|     acc = acc.concat(curr); | ||||
|     return acc; | ||||
|   }, []); | ||||
|  | ||||
|   console.log(reducedPromises); | ||||
|   await Promise.all(reducedPromises); | ||||
|   moveGroupChatToModel(skip + 50); | ||||
| } | ||||
|  | ||||
| module.exports = moveGroupChatToModel; | ||||
							
								
								
									
										107
									
								
								migrations/groups/reconcile-group-plan-members.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								migrations/groups/reconcile-group-plan-members.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | ||||
| import monk from 'monk'; | ||||
| import nconf from 'nconf'; | ||||
| import stripePayments from '../../website/server/libs/payments/stripe'; | ||||
|  | ||||
| /* | ||||
|  * Ensure that group plan billing is accurate by doing the following: | ||||
|  * 1. Correct the memberCount in all paid groups whose counts are wrong | ||||
|  * 2. Where the above uses Stripe, update their subscription counts in Stripe | ||||
|  * | ||||
|  * Provides output on what groups were fixed, which can be piped to CSV. | ||||
|  */ | ||||
|  | ||||
| const CONNECTION_STRING = nconf.get('MIGRATION_CONNECT_STRING'); | ||||
|  | ||||
| let dbGroups = monk(CONNECTION_STRING).get('groups', { castIds: false }); | ||||
| let dbUsers = monk(CONNECTION_STRING).get('users', { castIds: false }); | ||||
|  | ||||
| async function fixGroupPlanMembers () { | ||||
|   console.info('Group ID, Customer ID, Plan ID, Quantity, Recorded Member Count, Actual Member Count'); | ||||
|   let groupPlanCount = 0; | ||||
|   let fixedGroupCount = 0; | ||||
|  | ||||
|   dbGroups.find( | ||||
|     { | ||||
|       $and: | ||||
|         [ | ||||
|           {'purchased.plan.planId': {$ne: null}}, | ||||
|           {'purchased.plan.planId': {$ne: ''}}, | ||||
|           {'purchased.plan.customerId': {$ne: 'cus_9f0DV4g7WHRzpM'}}, // Demo groups | ||||
|           {'purchased.plan.customerId': {$ne: 'cus_9maalqDOFTrvqx'}}, | ||||
|         ], | ||||
|       $or: | ||||
|         [ | ||||
|           {'purchased.plan.dateTerminated': null}, | ||||
|           {'purchased.plan.dateTerminated': ''}, | ||||
|         ], | ||||
|     }, | ||||
|     { | ||||
|       fields: { | ||||
|         memberCount: 1, | ||||
|         'purchased.plan': 1, | ||||
|       }, | ||||
|     } | ||||
|   ).each(async (group, {close, pause, resume}) => { // eslint-disable-line no-unused-vars | ||||
|     pause(); | ||||
|     groupPlanCount++; | ||||
|  | ||||
|     const canonicalMemberCount = await dbUsers.count( | ||||
|       { | ||||
|         $or: | ||||
|           [ | ||||
|             {'party._id': group._id}, | ||||
|             {guilds: group._id}, | ||||
|           ], | ||||
|       } | ||||
|     ); | ||||
|     const incorrectMemberCount = group.memberCount !== canonicalMemberCount; | ||||
|  | ||||
|     const isMonthlyPlan = group.purchased.plan.planId === 'group_monthly'; | ||||
|     const quantityMismatch = group.purchased.plan.quantity !== group.memberCount + 2; | ||||
|     const incorrectQuantity = isMonthlyPlan && quantityMismatch; | ||||
|  | ||||
|     if (!incorrectMemberCount && !incorrectQuantity) { | ||||
|       resume(); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     console.info(`${group._id}, ${group.purchased.plan.customerId}, ${group.purchased.plan.planId}, ${group.purchased.plan.quantity}, ${group.memberCount}, ${canonicalMemberCount}`); | ||||
|  | ||||
|     const groupUpdate = await dbGroups.update( | ||||
|       { _id: group._id }, | ||||
|       { | ||||
|         $set: { | ||||
|           memberCount: canonicalMemberCount, | ||||
|         }, | ||||
|       } | ||||
|     ); | ||||
|  | ||||
|     if (!groupUpdate) return; | ||||
|  | ||||
|     fixedGroupCount++; | ||||
|     if (group.purchased.plan.paymentMethod === 'Stripe') { | ||||
|       await stripePayments.chargeForAdditionalGroupMember(group); | ||||
|       await dbGroups.update( | ||||
|         {_id: group._id}, | ||||
|         {$set: {'purchased.plan.quantity': canonicalMemberCount + 2}} | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (incorrectQuantity) { | ||||
|       await dbGroups.update( | ||||
|         {_id: group._id}, | ||||
|         {$set: {'purchased.plan.quantity': canonicalMemberCount + 2}} | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     resume(); | ||||
|   }).then(() => { | ||||
|     console.info(`Fixed ${fixedGroupCount} out of ${groupPlanCount} active Group Plans`); | ||||
|     return process.exit(0); | ||||
|   }).catch((err) => { | ||||
|     console.log(err); | ||||
|     return process.exit(1); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| module.exports = fixGroupPlanMembers; | ||||
| @@ -17,5 +17,5 @@ function setUpServer () { | ||||
| setUpServer(); | ||||
|  | ||||
| // Replace this with your migration | ||||
| const processUsers = require('./20180125_clean_new_notifications.js'); | ||||
| const processUsers = require('./tasks/habits-one-history-entry-per-day-challenges.js'); | ||||
| processUsers(); | ||||
|   | ||||
							
								
								
									
										138
									
								
								migrations/tasks/habits-one-history-entry-per-day-challenges.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								migrations/tasks/habits-one-history-entry-per-day-challenges.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | ||||
| // const migrationName = 'habits-one-history-entry-per-day'; | ||||
| // const authorName = 'paglias'; // in case script author needs to know when their ... | ||||
| // const authorUuid = 'ed4c688c-6652-4a92-9d03-a5a79844174a'; // ... own data is done | ||||
|  | ||||
| /* | ||||
|  * Iterates over all habits and condense multiple history entries for the same day into a single entry | ||||
|  */ | ||||
|  | ||||
| const monk = require('monk'); | ||||
| const _ = require('lodash'); | ||||
| const moment = require('moment'); | ||||
| const connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE | ||||
| const dbTasks = monk(connectionString).get('tasks', { castIds: false }); | ||||
|  | ||||
| function processChallengeHabits (lastId) { | ||||
|   let query = { | ||||
|     'challenge.id': {$exists: true}, | ||||
|     userId: {$exists: false}, | ||||
|     type: 'habit', | ||||
|   }; | ||||
|  | ||||
|   if (lastId) { | ||||
|     query._id = { | ||||
|       $gt: lastId, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   dbTasks.find(query, { | ||||
|     sort: {_id: 1}, | ||||
|     limit: 500, | ||||
|   }) | ||||
|     .then(updateChallengeHabits) | ||||
|     .catch((err) => { | ||||
|       console.log(err); | ||||
|       return exiting(1, `ERROR! ${  err}`); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| let progressCount = 1000; | ||||
| let count = 0; | ||||
|  | ||||
| function updateChallengeHabits (habits) { | ||||
|   if (!habits || habits.length === 0) { | ||||
|     console.warn('All appropriate challenge habits found and modified.'); | ||||
|     displayData(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   let habitsPromises = habits.map(updateChallengeHabit); | ||||
|   let lastHabit = habits[habits.length - 1]; | ||||
|  | ||||
|   return Promise.all(habitsPromises) | ||||
|     .then(() => { | ||||
|       return processChallengeHabits(lastHabit._id); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function updateChallengeHabit (habit) { | ||||
|   count++; | ||||
|  | ||||
|   if (habit && habit.history && habit.history.length > 0) { | ||||
|     // First remove missing entries | ||||
|     habit.history = habit.history.filter(entry => Boolean(entry)); | ||||
|  | ||||
|     habit.history = _.chain(habit.history) | ||||
|       // processes all entries to identify an up or down score | ||||
|       .forEach((entry, index) => { | ||||
|         if (index === 0) { // first entry doesn't have a previous one | ||||
|           // first value < 0 identifies a negative score as the first action | ||||
|           entry.scoreDirection = entry.value >= 0 ? 'up' : 'down'; | ||||
|         } else { | ||||
|           // could be missing if the previous entry was null and thus excluded | ||||
|           const previousEntry = habit.history[index - 1]; | ||||
|           const previousValue = previousEntry.value; | ||||
|  | ||||
|           entry.scoreDirection = entry.value > previousValue ? 'up' : 'down'; | ||||
|         } | ||||
|       }) | ||||
|       .groupBy(entry => { // group entries by aggregateBy | ||||
|         return moment(entry.date).format('YYYYMMDD'); | ||||
|       }) | ||||
|       .toPairs() // [key, entry] | ||||
|       .sortBy(([key]) => key) // sort by date | ||||
|       .map(keyEntryPair => { | ||||
|         let entries = keyEntryPair[1]; // 1 is entry, 0 is key | ||||
|         let scoredUp = 0; | ||||
|         let scoredDown = 0; | ||||
|  | ||||
|         entries.forEach(entry => { | ||||
|           if (entry.scoreDirection === 'up') { | ||||
|             scoredUp += 1; | ||||
|           } else { | ||||
|             scoredDown += 1; | ||||
|           } | ||||
|  | ||||
|           // delete the unnecessary scoreDirection and scoreNotes prop | ||||
|           delete entry.scoreDirection; | ||||
|           delete entry.scoreNotes; | ||||
|         }); | ||||
|  | ||||
|         return { | ||||
|           date: Number(entries[entries.length - 1].date), // keep last value | ||||
|           value: entries[entries.length - 1].value, // keep last value, | ||||
|           scoredUp, | ||||
|           scoredDown, | ||||
|         }; | ||||
|       }) | ||||
|       .value(); | ||||
|  | ||||
|     return dbTasks.update({_id: habit._id}, { | ||||
|       $set: {history: habit.history}, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   if (count % progressCount === 0) console.warn(`${count  } habits processed`); | ||||
| } | ||||
|  | ||||
| function displayData () { | ||||
|   console.warn(`\n${  count  } tasks processed\n`); | ||||
|   return exiting(0); | ||||
| } | ||||
|  | ||||
| function exiting (code, msg) { | ||||
|   code = code || 0; // 0 = success | ||||
|   if (code && !msg) { | ||||
|     msg = 'ERROR!'; | ||||
|   } | ||||
|   if (msg) { | ||||
|     if (code) { | ||||
|       console.error(msg); | ||||
|     } else      { | ||||
|       console.log(msg); | ||||
|     } | ||||
|   } | ||||
|   process.exit(code); | ||||
| } | ||||
|  | ||||
| module.exports = processChallengeHabits; | ||||
							
								
								
									
										163
									
								
								migrations/tasks/habits-one-history-entry-per-day-users.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								migrations/tasks/habits-one-history-entry-per-day-users.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| const migrationName = 'habits-one-history-entry-per-day'; | ||||
| const authorName = 'paglias'; // in case script author needs to know when their ... | ||||
| const authorUuid = 'ed4c688c-6652-4a92-9d03-a5a79844174a'; // ... own data is done | ||||
|  | ||||
| /* | ||||
|  * Iterates over all habits and condense multiple history entries for the same day into a single entry | ||||
|  */ | ||||
|  | ||||
| const monk = require('monk'); | ||||
| const _ = require('lodash'); | ||||
| const moment = require('moment'); | ||||
| const connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE | ||||
| const dbTasks = monk(connectionString).get('tasks', { castIds: false }); | ||||
| const dbUsers = monk(connectionString).get('users', { castIds: false }); | ||||
|  | ||||
| function processUsers (lastId) { | ||||
|   let query = { | ||||
|     migration: {$ne: migrationName}, | ||||
|   }; | ||||
|  | ||||
|   if (lastId) { | ||||
|     query._id = { | ||||
|       $gt: lastId, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   dbUsers.find(query, { | ||||
|     sort: {_id: 1}, | ||||
|     limit: 50, // just 50 users per time since we have to process all their habits as well | ||||
|     fields: ['_id', 'preferences.timezoneOffset', 'preferences.dayStart'], | ||||
|   }) | ||||
|     .then(updateUsers) | ||||
|     .catch((err) => { | ||||
|       console.log(err); | ||||
|       return exiting(1, `ERROR! ${  err}`); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| let progressCount = 1000; | ||||
| let count = 0; | ||||
|  | ||||
| function updateUsers (users) { | ||||
|   if (!users || users.length === 0) { | ||||
|     console.warn('All appropriate users and their tasks found and modified.'); | ||||
|     displayData(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   let usersPromises = users.map(updateUser); | ||||
|   let lastUser = users[users.length - 1]; | ||||
|  | ||||
|   return Promise.all(usersPromises) | ||||
|     .then(() => { | ||||
|       return processUsers(lastUser._id); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function updateHabit (habit, timezoneOffset, dayStart) { | ||||
|   if (habit && habit.history && habit.history.length > 0) { | ||||
|     // First remove missing entries | ||||
|     habit.history = habit.history.filter(entry => Boolean(entry)); | ||||
|  | ||||
|     habit.history = _.chain(habit.history) | ||||
|       // processes all entries to identify an up or down score | ||||
|       .forEach((entry, index) => { | ||||
|         if (index === 0) { // first entry doesn't have a previous one | ||||
|           // first value < 0 identifies a negative score as the first action | ||||
|           entry.scoreDirection = entry.value >= 0 ? 'up' : 'down'; | ||||
|         } else { | ||||
|           // could be missing if the previous entry was null and thus excluded | ||||
|           const previousEntry = habit.history[index - 1]; | ||||
|           const previousValue = previousEntry.value; | ||||
|  | ||||
|           entry.scoreDirection = entry.value > previousValue ? 'up' : 'down'; | ||||
|         } | ||||
|       }) | ||||
|       .groupBy(entry => { // group entries by aggregateBy | ||||
|         const entryDate = moment(entry.date).zone(timezoneOffset || 0); | ||||
|         if (entryDate.hour() < dayStart) entryDate.subtract(1, 'day'); | ||||
|         return entryDate.format('YYYYMMDD'); | ||||
|       }) | ||||
|       .toPairs() // [key, entry] | ||||
|       .sortBy(([key]) => key) // sort by date | ||||
|       .map(keyEntryPair => { | ||||
|         let entries = keyEntryPair[1]; // 1 is entry, 0 is key | ||||
|         let scoredUp = 0; | ||||
|         let scoredDown = 0; | ||||
|  | ||||
|         entries.forEach(entry => { | ||||
|           if (entry.scoreDirection === 'up') { | ||||
|             scoredUp += 1; | ||||
|           } else { | ||||
|             scoredDown += 1; | ||||
|           } | ||||
|  | ||||
|           // delete the unnecessary scoreDirection and scoreNotes prop | ||||
|           delete entry.scoreDirection; | ||||
|           delete entry.scoreNotes; | ||||
|         }); | ||||
|  | ||||
|         return { | ||||
|           date: Number(entries[entries.length - 1].date), // keep last value | ||||
|           value: entries[entries.length - 1].value, // keep last value, | ||||
|           scoredUp, | ||||
|           scoredDown, | ||||
|         }; | ||||
|       }) | ||||
|       .value(); | ||||
|  | ||||
|     return dbTasks.update({_id: habit._id}, { | ||||
|       $set: {history: habit.history}, | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|  | ||||
| function updateUser (user) { | ||||
|   count++; | ||||
|  | ||||
|   const timezoneOffset = user.preferences.timezoneOffset; | ||||
|   const dayStart = user.preferences.dayStart; | ||||
|  | ||||
|   if (count % progressCount === 0) console.warn(`${count  } ${  user._id}`); | ||||
|   if (user._id === authorUuid) console.warn(`${authorName  } being processed`); | ||||
|  | ||||
|   return dbTasks.find({ | ||||
|     type: 'habit', | ||||
|     userId: user._id, | ||||
|   }) | ||||
|     .then(habits => { | ||||
|       return Promise.all(habits.map(habit => updateHabit(habit, timezoneOffset, dayStart))); | ||||
|     }) | ||||
|     .then(() => { | ||||
|       return dbUsers.update({_id: user._id}, { | ||||
|         $set: {migration: migrationName}, | ||||
|       }); | ||||
|     }) | ||||
|     .catch((err) => { | ||||
|       console.log(err); | ||||
|       return exiting(1, `ERROR! ${  err}`); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function displayData () { | ||||
|   console.warn(`\n${  count  } tasks processed\n`); | ||||
|   return exiting(0); | ||||
| } | ||||
|  | ||||
| function exiting (code, msg) { | ||||
|   code = code || 0; // 0 = success | ||||
|   if (code && !msg) { | ||||
|     msg = 'ERROR!'; | ||||
|   } | ||||
|   if (msg) { | ||||
|     if (code) { | ||||
|       console.error(msg); | ||||
|     } else      { | ||||
|       console.log(msg); | ||||
|     } | ||||
|   } | ||||
|   process.exit(code); | ||||
| } | ||||
|  | ||||
| module.exports = processUsers; | ||||
| @@ -7,7 +7,7 @@ let authorUuid = '7f14ed62-5408-4e1b-be83-ada62d504931'; // ... own data is done | ||||
|  */ | ||||
|  | ||||
| let monk = require('monk'); | ||||
| let connectionString = 'mongodb://sabrecat:z8e8jyRA8CTofMQ@ds013393-a0.mlab.com:13393/habitica?auto_reconnect=true'; | ||||
| let connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; | ||||
| let dbTasks = monk(connectionString).get('tasks', { castIds: false }); | ||||
|  | ||||
| function processTasks (lastId) { | ||||
|   | ||||
| @@ -1,15 +1,17 @@ | ||||
| const migrationName = 'mystery-items-201802.js'; // Update per month | ||||
| import monk from 'monk'; | ||||
| import nconf from 'nconf'; | ||||
|  | ||||
| const migrationName = 'mystery-items-201806.js'; // Update per month | ||||
| const authorName = 'Sabe'; // in case script author needs to know when their ... | ||||
| const authorUuid = '7f14ed62-5408-4e1b-be83-ada62d504931'; // ... own data is done | ||||
|  | ||||
| /* | ||||
|  * Award this month's mystery items to subscribers | ||||
|  */ | ||||
| const MYSTERY_ITEMS = ['back_mystery_201803', 'head_mystery_201803']; | ||||
| const connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE | ||||
| const MYSTERY_ITEMS = ['armor_mystery_201806', 'head_mystery_201806']; | ||||
| const CONNECTION_STRING = nconf.get('MIGRATION_CONNECT_STRING'); | ||||
|  | ||||
| let monk = require('monk'); | ||||
| let dbUsers = monk(connectionString).get('users', { castIds: false }); | ||||
| let dbUsers = monk(CONNECTION_STRING).get('users', { castIds: false }); | ||||
| let UserNotification = require('../../website/server/models/userNotification').model; | ||||
|  | ||||
| function processUsers (lastId) { | ||||
|   | ||||
							
								
								
									
										109
									
								
								migrations/users/remove-social-users-extra-data.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								migrations/users/remove-social-users-extra-data.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,109 @@ | ||||
| const migrationName = 'remove-social-users-extra-data.js'; | ||||
| const authorName = 'paglias'; // in case script author needs to know when their ... | ||||
| const authorUuid = 'ed4c688c-6652-4a92-9d03-a5a79844174a'; // ... own data is done | ||||
|  | ||||
| /* | ||||
|  * Remove not needed data from social profiles | ||||
|  */ | ||||
| const connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE | ||||
|  | ||||
| const monk = require('monk'); | ||||
| const dbUsers = monk(connectionString).get('users', { castIds: false }); | ||||
|  | ||||
| function processUsers (lastId) { | ||||
|   // specify a query to limit the affected users (empty for all users): | ||||
|   let query = { | ||||
|     migration: {$ne: migrationName}, | ||||
|     $or: [ | ||||
|       { 'auth.facebook.id': { $exists: true } }, | ||||
|       { 'auth.google.id': { $exists: true } }, | ||||
|     ], | ||||
|   }; | ||||
|  | ||||
|   if (lastId) { | ||||
|     query._id = { | ||||
|       $gt: lastId, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   dbUsers.find(query, { | ||||
|     sort: {_id: 1}, | ||||
|     limit: 250, | ||||
|   }) | ||||
|     .then(updateUsers) | ||||
|     .catch((err) => { | ||||
|       console.log(err); | ||||
|       return exiting(1, `ERROR! ${  err}`); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| let progressCount = 1000; | ||||
| let count = 0; | ||||
|  | ||||
| function updateUsers (users) { | ||||
|   if (!users || users.length === 0) { | ||||
|     console.warn('All appropriate users found and modified.'); | ||||
|     displayData(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   let userPromises = users.map(updateUser); | ||||
|   let lastUser = users[users.length - 1]; | ||||
|  | ||||
|   return Promise.all(userPromises) | ||||
|     .then(() => { | ||||
|       processUsers(lastUser._id); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| function updateUser (user) { | ||||
|   count++; | ||||
|  | ||||
|   const isFacebook = user.auth.facebook && user.auth.facebook.id; | ||||
|   const isGoogle = user.auth.google && user.auth.google.id; | ||||
|  | ||||
|   const update = { $set: {} }; | ||||
|  | ||||
|   if (isFacebook) { | ||||
|     update.$set['auth.facebook'] = { | ||||
|       id: user.auth.facebook.id, | ||||
|       emails: user.auth.facebook.emails, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   if (isGoogle) { | ||||
|     update.$set['auth.google'] = { | ||||
|       id: user.auth.google.id, | ||||
|       emails: user.auth.google.emails, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   dbUsers.update({ | ||||
|     _id: user._id, | ||||
|   }, update); | ||||
|  | ||||
|   if (count % progressCount === 0) console.warn(`${count  } ${  user._id}`); | ||||
|   if (user._id === authorUuid) console.warn(`${authorName  } processed`); | ||||
| } | ||||
|  | ||||
| function displayData () { | ||||
|   console.warn(`\n${  count  } users processed\n`); | ||||
|   return exiting(0); | ||||
| } | ||||
|  | ||||
| function exiting (code, msg) { | ||||
|   code = code || 0; // 0 = success | ||||
|   if (code && !msg) { | ||||
|     msg = 'ERROR!'; | ||||
|   } | ||||
|   if (msg) { | ||||
|     if (code) { | ||||
|       console.error(msg); | ||||
|     } else      { | ||||
|       console.log(msg); | ||||
|     } | ||||
|   } | ||||
|   process.exit(code); | ||||
| } | ||||
|  | ||||
| module.exports = processUsers; | ||||
| @@ -1,4 +1,4 @@ | ||||
| let migrationName = '20180102_takeThis.js'; // Update per month
 | ||||
| let migrationName = '20180702_takeThis.js'; // Update per month
 | ||||
| let authorName = 'Sabe'; // in case script author needs to know when their ...
 | ||||
| let authorUuid = '7f14ed62-5408-4e1b-be83-ada62d504931'; // ... own data is done
 | ||||
| 
 | ||||
| @@ -6,15 +6,16 @@ let authorUuid = '7f14ed62-5408-4e1b-be83-ada62d504931'; // ... own data is done | ||||
|  * Award Take This ladder items to participants in this month's challenge | ||||
|  */ | ||||
| 
 | ||||
| let monk = require('monk'); | ||||
| let connectionString = 'mongodb://localhost:27017/habitrpg?auto_reconnect=true'; // FOR TEST DATABASE
 | ||||
| let dbUsers = monk(connectionString).get('users', { castIds: false }); | ||||
| import monk from 'monk'; | ||||
| import nconf from 'nconf'; | ||||
| const CONNECTION_STRING = nconf.get('MIGRATION_CONNECT_STRING'); // FOR TEST DATABASE
 | ||||
| let dbUsers = monk(CONNECTION_STRING).get('users', { castIds: false }); | ||||
| 
 | ||||
| function processUsers (lastId) { | ||||
|   // specify a query to limit the affected users (empty for all users):
 | ||||
|   let query = { | ||||
|     migration: {$ne: migrationName}, | ||||
|     challenges: {$in: ['5f70ce5b-2d82-4114-8e44-ca65615aae62']}, // Update per month
 | ||||
|     challenges: {$in: ['f0481f95-1dde-4ae7-a876-d19502a45d61']}, // Update per month
 | ||||
|   }; | ||||
| 
 | ||||
|   if (lastId) { | ||||
							
								
								
									
										13019
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										13019
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										119
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										119
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,48 +1,49 @@ | ||||
| { | ||||
|   "name": "habitica", | ||||
|   "description": "A habit tracker app which treats your goals like a Role Playing Game.", | ||||
|   "version": "4.34.2", | ||||
|   "version": "4.52.2", | ||||
|   "main": "./website/server/index.js", | ||||
|   "dependencies": { | ||||
|     "@slack/client": "^3.8.1", | ||||
|     "accepts": "^1.3.5", | ||||
|     "amazon-payments": "^0.2.6", | ||||
|     "amazon-payments": "^0.2.7", | ||||
|     "amplitude": "^3.5.0", | ||||
|     "apidoc": "^0.17.5", | ||||
|     "autoprefixer": "^8.1.0", | ||||
|     "aws-sdk": "^2.211.0", | ||||
|     "autoprefixer": "^8.5.0", | ||||
|     "aws-sdk": "^2.239.1", | ||||
|     "apn": "^2.2.0", | ||||
|     "axios": "^0.18.0", | ||||
|     "axios-progress-bar": "^1.1.8", | ||||
|     "babel-core": "^6.0.0", | ||||
|     "babel-eslint": "^8.2.2", | ||||
|     "axios-progress-bar": "^1.2.0", | ||||
|     "babel-core": "^6.26.3", | ||||
|     "babel-eslint": "^8.2.3", | ||||
|     "babel-loader": "^7.1.4", | ||||
|     "babel-plugin-syntax-async-functions": "^6.13.0", | ||||
|     "babel-plugin-syntax-dynamic-import": "^6.18.0", | ||||
|     "babel-plugin-transform-es2015-modules-commonjs": "^6.26.0", | ||||
|     "babel-plugin-transform-es2015-modules-commonjs": "^6.26.2", | ||||
|     "babel-plugin-transform-object-rest-spread": "^6.16.0", | ||||
|     "babel-plugin-transform-regenerator": "^6.16.1", | ||||
|     "babel-polyfill": "^6.6.1", | ||||
|     "babel-preset-es2015": "^6.6.0", | ||||
|     "babel-register": "^6.6.0", | ||||
|     "babel-runtime": "^6.11.6", | ||||
|     "bcrypt": "^1.0.2", | ||||
|     "body-parser": "^1.15.0", | ||||
|     "bootstrap": "^4.0.0", | ||||
|     "bootstrap-vue": "^2.0.0-rc.2", | ||||
|     "bcrypt": "^2.0.0", | ||||
|     "body-parser": "^1.18.3", | ||||
|     "bootstrap": "^4.1.1", | ||||
|     "bootstrap-vue": "^2.0.0-rc.9", | ||||
|     "compression": "^1.7.2", | ||||
|     "cookie-session": "^1.2.0", | ||||
|     "coupon-code": "^0.4.5", | ||||
|     "cross-env": "^5.1.4", | ||||
|     "cross-env": "^5.1.5", | ||||
|     "css-loader": "^0.28.11", | ||||
|     "csv-stringify": "^2.0.4", | ||||
|     "csv-stringify": "^2.1.0", | ||||
|     "cwait": "^1.1.1", | ||||
|     "domain-middleware": "~0.1.0", | ||||
|     "express": "^4.16.3", | ||||
|     "express-basic-auth": "^1.1.4", | ||||
|     "express-validator": "^5.0.3", | ||||
|     "express-basic-auth": "^1.1.5", | ||||
|     "express-validator": "^5.2.0", | ||||
|     "extract-text-webpack-plugin": "^3.0.2", | ||||
|     "glob": "^7.1.2", | ||||
|     "got": "^8.3.0", | ||||
|     "got": "^8.3.1", | ||||
|     "gulp": "^4.0.0", | ||||
|     "gulp-babel": "^7.0.1", | ||||
|     "gulp-imagemin": "^4.1.0", | ||||
| @@ -50,64 +51,63 @@ | ||||
|     "gulp.spritesmith": "^6.9.0", | ||||
|     "habitica-markdown": "^1.3.0", | ||||
|     "hellojs": "^1.15.1", | ||||
|     "html-webpack-plugin": "^3.0.0", | ||||
|     "html-webpack-plugin": "^3.2.0", | ||||
|     "image-size": "^0.6.2", | ||||
|     "in-app-purchase": "^1.8.9", | ||||
|     "intro.js": "^2.6.0", | ||||
|     "in-app-purchase": "^1.9.4", | ||||
|     "intro.js": "^2.9.3", | ||||
|     "jquery": ">=3.0.0", | ||||
|     "js2xmlparser": "^3.0.0", | ||||
|     "lodash": "^4.17.4", | ||||
|     "memwatch-next": "^0.3.0", | ||||
|     "lodash": "^4.17.10", | ||||
|     "merge-stream": "^1.0.0", | ||||
|     "method-override": "^2.3.5", | ||||
|     "moment": "^2.21.0", | ||||
|     "moment": "^2.22.1", | ||||
|     "moment-recur": "^1.0.7", | ||||
|     "mongoose": "^5.0.10", | ||||
|     "mongoose": "^5.1.2", | ||||
|     "morgan": "^1.7.0", | ||||
|     "nconf": "^0.10.0", | ||||
|     "node-gcm": "^0.14.4", | ||||
|     "node-sass": "^4.8.2", | ||||
|     "nodemailer": "^4.6.3", | ||||
|     "ora": "^2.0.0", | ||||
|     "node-sass": "^4.9.0", | ||||
|     "nodemailer": "^4.6.4", | ||||
|     "ora": "^2.1.0", | ||||
|     "pageres": "^4.1.1", | ||||
|     "passport": "^0.4.0", | ||||
|     "passport-facebook": "^2.0.0", | ||||
|     "passport-google-oauth20": "1.0.0", | ||||
|     "paypal-ipn": "3.0.0", | ||||
|     "paypal-rest-sdk": "^1.8.1", | ||||
|     "popper.js": "^1.14.1", | ||||
|     "popper.js": "^1.14.3", | ||||
|     "postcss-easy-import": "^3.0.0", | ||||
|     "ps-tree": "^1.0.0", | ||||
|     "pug": "^2.0.1", | ||||
|     "push-notify": "git://github.com/habitrpg/push-notify.git#6bc2b5fdb1bdc9649b9ec1964d79ca50187fc8a9", | ||||
|     "pug": "^2.0.3", | ||||
|     "pusher": "^1.3.0", | ||||
|     "rimraf": "^2.4.3", | ||||
|     "sass-loader": "^6.0.7", | ||||
|     "shelljs": "^0.8.1", | ||||
|     "stackimpact": "^1.2.1", | ||||
|     "stripe": "^5.5.0", | ||||
|     "superagent": "^3.4.3", | ||||
|     "sass-loader": "^7.0.0", | ||||
|     "shelljs": "^0.8.2", | ||||
|     "stripe": "^5.9.0", | ||||
|     "superagent": "^3.8.3", | ||||
|     "svg-inline-loader": "^0.8.0", | ||||
|     "svg-url-loader": "^2.3.2", | ||||
|     "svgo": "^1.0.5", | ||||
|     "svgo-loader": "^2.1.0", | ||||
|     "universal-analytics": "^0.4.16", | ||||
|     "update": "^0.7.4", | ||||
|     "upgrade": "^1.1.0", | ||||
|     "url-loader": "^1.0.0", | ||||
|     "useragent": "^2.1.9", | ||||
|     "uuid": "^3.0.1", | ||||
|     "validator": "^9.4.1", | ||||
|     "vinyl-buffer": "^1.0.1", | ||||
|     "vue": "^2.5.16", | ||||
|     "vue-loader": "^14.2.1", | ||||
|     "vue-loader": "^14.2.2", | ||||
|     "vue-mugen-scroll": "^0.2.1", | ||||
|     "vue-router": "^3.0.0", | ||||
|     "vue-style-loader": "^4.0.2", | ||||
|     "vue-style-loader": "^4.1.0", | ||||
|     "vue-template-compiler": "^2.5.16", | ||||
|     "vuedraggable": "^2.15.0", | ||||
|     "vuejs-datepicker": "git://github.com/habitrpg/vuejs-datepicker.git#5d237615463a84a23dd6f3f77c6ab577d68593ec", | ||||
|     "webpack": "^3.11.0", | ||||
|     "webpack": "^3.12.0", | ||||
|     "webpack-merge": "^4.0.0", | ||||
|     "winston": "^2.4.1", | ||||
|     "winston": "^2.4.2", | ||||
|     "winston-loggly-bulk": "^2.0.2", | ||||
|     "xml2js": "^0.4.4" | ||||
|   }, | ||||
| @@ -121,9 +121,11 @@ | ||||
|     "test": "npm run lint && gulp test && gulp apidoc", | ||||
|     "test:build": "gulp test:prepare:build", | ||||
|     "test:api-v3": "gulp test:api-v3", | ||||
|     "test:api-v3:unit": "gulp test:api-v3:unit", | ||||
|     "test:api:unit": "gulp test:api:unit", | ||||
|     "test:api-v3:integration": "gulp test:api-v3:integration", | ||||
|     "test:api-v3:integration:separate-server": "NODE_ENV=test gulp test:api-v3:integration:separate-server", | ||||
|     "test:api-v4:integration": "gulp test:api-v4:integration", | ||||
|     "test:api-v4:integration:separate-server": "NODE_ENV=test gulp test:api-v4:integration:separate-server", | ||||
|     "test:sanity": "istanbul cover --dir coverage/sanity --report lcovonly node_modules/mocha/bin/_mocha -- test/sanity --recursive", | ||||
|     "test:common": "istanbul cover --dir coverage/common --report lcovonly node_modules/mocha/bin/_mocha -- test/common --recursive", | ||||
|     "test:content": "istanbul cover --dir coverage/content --report lcovonly node_modules/mocha/bin/_mocha -- test/content --recursive", | ||||
| @@ -141,53 +143,54 @@ | ||||
|     "apidoc": "gulp apidoc" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@vue/test-utils": "^1.0.0-beta.12", | ||||
|     "@vue/test-utils": "^1.0.0-beta.16", | ||||
|     "babel-plugin-istanbul": "^4.1.6", | ||||
|     "babel-plugin-syntax-object-rest-spread": "^6.13.0", | ||||
|     "chai": "^4.1.2", | ||||
|     "chai-as-promised": "^7.1.1", | ||||
|     "chalk": "^2.3.2", | ||||
|     "chromedriver": "^2.36.0", | ||||
|     "chalk": "^2.4.1", | ||||
|     "chromedriver": "^2.38.3", | ||||
|     "connect-history-api-fallback": "^1.1.0", | ||||
|     "coveralls": "^3.0.0", | ||||
|     "coveralls": "^3.0.1", | ||||
|     "cross-spawn": "^6.0.5", | ||||
|     "eslint": "^4.19.0", | ||||
|     "eslint": "^4.19.1", | ||||
|     "eslint-config-habitrpg": "^4.0.0", | ||||
|     "eslint-friendly-formatter": "^4.0.0", | ||||
|     "eslint-friendly-formatter": "^4.0.1", | ||||
|     "eslint-loader": "^2.0.0", | ||||
|     "eslint-plugin-html": "^4.0.2", | ||||
|     "eslint-plugin-html": "^4.0.3", | ||||
|     "eslint-plugin-mocha": "^5.0.0", | ||||
|     "eventsource-polyfill": "^0.9.6", | ||||
|     "expect.js": "^0.3.1", | ||||
|     "http-proxy-middleware": "^0.18.0", | ||||
|     "istanbul": "^1.1.0-alpha.1", | ||||
|     "karma": "^2.0.0", | ||||
|     "karma": "^2.0.2", | ||||
|     "karma-babel-preprocessor": "^7.0.0", | ||||
|     "karma-chai-plugins": "^0.9.0", | ||||
|     "karma-chrome-launcher": "^2.2.0", | ||||
|     "karma-coverage": "^1.1.1", | ||||
|     "karma-coverage": "^1.1.2", | ||||
|     "karma-mocha": "^1.3.0", | ||||
|     "karma-mocha-reporter": "^2.2.5", | ||||
|     "karma-sinon-chai": "^1.3.3", | ||||
|     "karma-sinon-chai": "^1.3.4", | ||||
|     "karma-sinon-stub-promise": "^1.0.0", | ||||
|     "karma-sourcemap-loader": "^0.3.7", | ||||
|     "karma-spec-reporter": "0.0.32", | ||||
|     "karma-webpack": "^3.0.0", | ||||
|     "lcov-result-merger": "^2.0.0", | ||||
|     "mocha": "^5.0.4", | ||||
|     "monk": "^6.0.5", | ||||
|     "nightwatch": "^0.9.20", | ||||
|     "puppeteer": "^1.2.0", | ||||
|     "mocha": "^5.1.1", | ||||
|     "monk": "^6.0.6", | ||||
|     "nightwatch": "^0.9.21", | ||||
|     "puppeteer": "^1.4.0", | ||||
|     "require-again": "^2.0.0", | ||||
|     "selenium-server": "^3.11.0", | ||||
|     "sinon": "^4.4.5", | ||||
|     "selenium-server": "^3.12.0", | ||||
|     "sinon": "^4.5.0", | ||||
|     "sinon-chai": "^3.0.0", | ||||
|     "sinon-stub-promise": "^4.0.0", | ||||
|     "webpack-bundle-analyzer": "^2.11.1", | ||||
|     "webpack-bundle-analyzer": "^2.12.0", | ||||
|     "webpack-dev-middleware": "^2.0.5", | ||||
|     "webpack-hot-middleware": "^2.21.2" | ||||
|     "webpack-hot-middleware": "^2.22.2" | ||||
|   }, | ||||
|   "optionalDependencies": { | ||||
|     "memwatch-next": "^0.3.0", | ||||
|     "node-rdkafka": "^2.3.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import analyticsService from '../../../../../website/server/libs/analyticsService'; | ||||
| import analyticsService from '../../../../website/server/libs/analyticsService'; | ||||
| import Amplitude from 'amplitude'; | ||||
| import { Visitor } from 'universal-analytics'; | ||||
| 
 | ||||
| @@ -1,19 +1,19 @@ | ||||
| import apiMessages from '../../../../../website/server/libs/apiMessages'; | ||||
| import apiError from '../../../../website/server/libs/apiError'; | ||||
| 
 | ||||
| describe('API Messages', () => { | ||||
|   const message = 'Only public guilds support pagination.'; | ||||
|   it('returns an API message', () => { | ||||
|     expect(apiMessages('guildsOnlyPaginate')).to.equal(message); | ||||
|     expect(apiError('guildsOnlyPaginate')).to.equal(message); | ||||
|   }); | ||||
| 
 | ||||
|   it('throws if the API message does not exist', () => { | ||||
|     expect(() => apiMessages('iDoNotExist')).to.throw; | ||||
|     expect(() => apiError('iDoNotExist')).to.throw; | ||||
|   }); | ||||
| 
 | ||||
|   it('clones the passed variables', () => { | ||||
|     let vars = {a: 1}; | ||||
|     sandbox.stub(_, 'clone').returns({}); | ||||
|     apiMessages('guildsOnlyPaginate', vars); | ||||
|     apiError('guildsOnlyPaginate', vars); | ||||
|     expect(_.clone).to.have.been.calledOnce; | ||||
|     expect(_.clone).to.have.been.calledWith(vars); | ||||
|   }); | ||||
| @@ -22,7 +22,7 @@ describe('API Messages', () => { | ||||
|     let vars = {a: 1}; | ||||
|     let stub = sinon.stub().returns('string'); | ||||
|     sandbox.stub(_, 'template').returns(stub); | ||||
|     apiMessages('guildsOnlyPaginate', vars); | ||||
|     apiError('guildsOnlyPaginate', vars); | ||||
|     expect(_.template).to.have.been.calledOnce; | ||||
|     expect(_.template).to.have.been.calledWith(message); | ||||
|     expect(stub).to.have.been.calledOnce; | ||||
| @@ -1,4 +1,4 @@ | ||||
| import baseModel from '../../../../../website/server/libs/baseModel'; | ||||
| import baseModel from '../../../../website/server/libs/baseModel'; | ||||
| import mongoose from 'mongoose'; | ||||
| 
 | ||||
| describe('Base model plugin', () => { | ||||
| @@ -1,7 +1,7 @@ | ||||
| import mongoose from 'mongoose'; | ||||
| import { | ||||
|   removeFromArray, | ||||
| } from '../../../../../website/server/libs/collectionManipulators'; | ||||
| } from '../../../../website/server/libs/collectionManipulators'; | ||||
| 
 | ||||
| describe('Collection Manipulators', () => { | ||||
|   describe('removeFromArray', () => { | ||||
| @@ -2,17 +2,18 @@ | ||||
| import moment from 'moment'; | ||||
| import nconf from 'nconf'; | ||||
| import requireAgain from 'require-again'; | ||||
| import { recoverCron, cron } from '../../../../../website/server/libs/cron'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../../website/server/models/task'; | ||||
| import common from '../../../../../website/common'; | ||||
| import analytics from '../../../../../website/server/libs/analyticsService'; | ||||
| import { recoverCron, cron } from '../../../../website/server/libs/cron'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../website/server/models/task'; | ||||
| import common from '../../../../website/common'; | ||||
| import analytics from '../../../../website/server/libs/analyticsService'; | ||||
| 
 | ||||
| // const scoreTask = common.ops.scoreTask;
 | ||||
| 
 | ||||
| let pathToCronLib = '../../../../../website/server/libs/cron'; | ||||
| let pathToCronLib = '../../../../website/server/libs/cron'; | ||||
| 
 | ||||
| describe('cron', () => { | ||||
|   let clock = null; | ||||
|   let user; | ||||
|   let tasksByType = {habits: [], dailys: [], todos: [], rewards: []}; | ||||
|   let daysMissed = 0; | ||||
| @@ -23,7 +24,7 @@ describe('cron', () => { | ||||
|         local: { | ||||
|           username: 'username', | ||||
|           lowerCaseUsername: 'username', | ||||
|           email: 'email@email.email', | ||||
|           email: 'email@example.com', | ||||
|           salt: 'salt', | ||||
|           hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|         }, | ||||
| @@ -34,6 +35,8 @@ describe('cron', () => { | ||||
|   }); | ||||
| 
 | ||||
|   afterEach(() => { | ||||
|     if (clock !== null) | ||||
|       clock.restore(); | ||||
|     analytics.track.restore(); | ||||
|   }); | ||||
| 
 | ||||
| @@ -82,14 +85,12 @@ describe('cron', () => { | ||||
|     }); | ||||
| 
 | ||||
|     it('does not reset plan.gemsBought within the month', () => { | ||||
|       let clock = sinon.useFakeTimers(moment().startOf('month').add(2, 'days').unix()); | ||||
|       clock = sinon.useFakeTimers(moment().startOf('month').add(2, 'days').toDate()); | ||||
|       user.purchased.plan.dateUpdated = moment().startOf('month').toDate(); | ||||
| 
 | ||||
|       user.purchased.plan.gemsBought = 10; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.gemsBought).to.equal(10); | ||||
| 
 | ||||
|       clock.restore(); | ||||
|     }); | ||||
| 
 | ||||
|     it('resets plan.dateUpdated on a new month', () => { | ||||
| @@ -117,21 +118,6 @@ describe('cron', () => { | ||||
|       expect(user.purchased.plan.consecutive.offset).to.equal(1); | ||||
|     }); | ||||
| 
 | ||||
|     it('increments plan.consecutive.trinkets when user has reached a month that is a multiple of 3', () => { | ||||
|       user.purchased.plan.consecutive.count = 5; | ||||
|       user.purchased.plan.consecutive.offset = 1; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|       expect(user.purchased.plan.consecutive.offset).to.equal(0); | ||||
|     }); | ||||
| 
 | ||||
|     it('increments plan.consecutive.trinkets multiple times if user has been absent with continuous subscription', () => { | ||||
|       user.purchased.plan.dateUpdated = moment().subtract(6, 'months').toDate(); | ||||
|       user.purchased.plan.consecutive.count = 5; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|     }); | ||||
| 
 | ||||
|     it('does not award unearned plan.consecutive.trinkets if subscription ended during an absence', () => { | ||||
|       user.purchased.plan.dateUpdated = moment().subtract(6, 'months').toDate(); | ||||
|       user.purchased.plan.dateTerminated = moment().subtract(3, 'months').toDate(); | ||||
| @@ -143,21 +129,6 @@ describe('cron', () => { | ||||
|       expect(user.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|     }); | ||||
| 
 | ||||
|     it('increments plan.consecutive.gemCapExtra when user has reached a month that is a multiple of 3', () => { | ||||
|       user.purchased.plan.consecutive.count = 5; | ||||
|       user.purchased.plan.consecutive.offset = 1; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       expect(user.purchased.plan.consecutive.offset).to.equal(0); | ||||
|     }); | ||||
| 
 | ||||
|     it('increments plan.consecutive.gemCapExtra multiple times if user has been absent with continuous subscription', () => { | ||||
|       user.purchased.plan.dateUpdated = moment().subtract(6, 'months').toDate(); | ||||
|       user.purchased.plan.consecutive.count = 5; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|     }); | ||||
| 
 | ||||
|     it('does not increment plan.consecutive.gemCapExtra when user has reached the gemCap limit', () => { | ||||
|       user.purchased.plan.consecutive.gemCapExtra = 25; | ||||
|       user.purchased.plan.consecutive.count = 5; | ||||
| @@ -184,6 +155,427 @@ describe('cron', () => { | ||||
|       expect(user.purchased.plan.consecutive.count).to.equal(0); | ||||
|       expect(user.purchased.plan.consecutive.offset).to.equal(0); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 1-month recurring subscription', () => { | ||||
|       // create a user that will be used for all of these tests without a reset before each
 | ||||
|       let user1 = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username1', | ||||
|             lowerCaseUsername: 'username1', | ||||
|             email: 'email1@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user1 has a 1-month recurring subscription starting today
 | ||||
|       user1.purchased.plan.customerId = 'subscribedId'; | ||||
|       user1.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user1.purchased.plan.planId = 'basic'; | ||||
|       user1.purchased.plan.consecutive.count = 0; | ||||
|       user1.purchased.plan.consecutive.offset = 0; | ||||
|       user1.purchased.plan.consecutive.trinkets = 0; | ||||
|       user1.purchased.plan.consecutive.gemCapExtra = 0; | ||||
| 
 | ||||
|       it('does not increment consecutive benefits after the first month', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         // Add 1 month to simulate what happens a month after the subscription was created.
 | ||||
|         // Add 2 days so that we're sure we're not affected by any start-of-month effects e.g., from time zone oddness.
 | ||||
|         cron({user: user1, tasksByType, daysMissed, analytics}); | ||||
|         expect(user1.purchased.plan.consecutive.count).to.equal(1); | ||||
|         expect(user1.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.trinkets).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.gemCapExtra).to.equal(0); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits after the second month', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(2, 'months').add(2, 'days').toDate()); | ||||
|         // Add 1 month to simulate what happens a month after the subscription was created.
 | ||||
|         // Add 2 days so that we're sure we're not affected by any start-of-month effects e.g., from time zone oddness.
 | ||||
|         cron({user: user1, tasksByType, daysMissed, analytics}); | ||||
|         expect(user1.purchased.plan.consecutive.count).to.equal(2); | ||||
|         expect(user1.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.trinkets).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.gemCapExtra).to.equal(0); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits after the third month', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(3, 'months').add(2, 'days').toDate()); | ||||
|         // Add 1 month to simulate what happens a month after the subscription was created.
 | ||||
|         // Add 2 days so that we're sure we're not affected by any start-of-month effects e.g., from time zone oddness.
 | ||||
|         cron({user: user1, tasksByType, daysMissed, analytics}); | ||||
|         expect(user1.purchased.plan.consecutive.count).to.equal(3); | ||||
|         expect(user1.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user1.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits after the fourth month', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(4, 'months').add(2, 'days').toDate()); | ||||
|         // Add 1 month to simulate what happens a month after the subscription was created.
 | ||||
|         // Add 2 days so that we're sure we're not affected by any start-of-month effects e.g., from time zone oddness.
 | ||||
|         cron({user: user1, tasksByType, daysMissed, analytics}); | ||||
|         expect(user1.purchased.plan.consecutive.count).to.equal(4); | ||||
|         expect(user1.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user1.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits correctly if user has been absent with continuous subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(10, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user1, tasksByType, daysMissed, analytics}); | ||||
|         expect(user1.purchased.plan.consecutive.count).to.equal(10); | ||||
|         expect(user1.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user1.purchased.plan.consecutive.trinkets).to.equal(3); | ||||
|         expect(user1.purchased.plan.consecutive.gemCapExtra).to.equal(15); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 3-month recurring subscription', () => { | ||||
|       let user3 = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username3', | ||||
|             lowerCaseUsername: 'username3', | ||||
|             email: 'email3@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user3 has a 3-month recurring subscription starting today
 | ||||
|       user3.purchased.plan.customerId = 'subscribedId'; | ||||
|       user3.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user3.purchased.plan.planId = 'basic_3mo'; | ||||
|       user3.purchased.plan.consecutive.count = 0; | ||||
|       user3.purchased.plan.consecutive.offset = 3; | ||||
|       user3.purchased.plan.consecutive.trinkets = 1; | ||||
|       user3.purchased.plan.consecutive.gemCapExtra = 5; | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the first month of the first paid period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the middle of the period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(2, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the final month of the period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(3, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(3); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the second paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(4, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(4); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the second month of the second period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(5, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(5); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(1); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the final month of the second period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(6, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(6); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the third paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(7, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(7); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(3); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(15); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits correctly if user has been absent with continuous subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(10, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3.purchased.plan.consecutive.count).to.equal(10); | ||||
|         expect(user3.purchased.plan.consecutive.offset).to.equal(2); | ||||
|         expect(user3.purchased.plan.consecutive.trinkets).to.equal(4); | ||||
|         expect(user3.purchased.plan.consecutive.gemCapExtra).to.equal(20); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 6-month recurring subscription', () => { | ||||
|       let user6 = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username6', | ||||
|             lowerCaseUsername: 'username6', | ||||
|             email: 'email6@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user6 has a 6-month recurring subscription starting today
 | ||||
|       user6.purchased.plan.customerId = 'subscribedId'; | ||||
|       user6.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user6.purchased.plan.planId = 'google_6mo'; | ||||
|       user6.purchased.plan.consecutive.count = 0; | ||||
|       user6.purchased.plan.consecutive.offset = 6; | ||||
|       user6.purchased.plan.consecutive.trinkets = 2; | ||||
|       user6.purchased.plan.consecutive.gemCapExtra = 10; | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the first month of the first paid period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6.purchased.plan.consecutive.count).to.equal(1); | ||||
|         expect(user6.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|         expect(user6.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the final month of the period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(6, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6.purchased.plan.consecutive.count).to.equal(6); | ||||
|         expect(user6.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user6.purchased.plan.consecutive.trinkets).to.equal(2); | ||||
|         expect(user6.purchased.plan.consecutive.gemCapExtra).to.equal(10); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the second paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(7, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6.purchased.plan.consecutive.count).to.equal(7); | ||||
|         expect(user6.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6.purchased.plan.consecutive.trinkets).to.equal(4); | ||||
|         expect(user6.purchased.plan.consecutive.gemCapExtra).to.equal(20); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the third paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(13, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6.purchased.plan.consecutive.count).to.equal(13); | ||||
|         expect(user6.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6.purchased.plan.consecutive.trinkets).to.equal(6); | ||||
|         expect(user6.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits correctly if user has been absent with continuous subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(19, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6.purchased.plan.consecutive.count).to.equal(19); | ||||
|         expect(user6.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6.purchased.plan.consecutive.trinkets).to.equal(8); | ||||
|         expect(user6.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 12-month recurring subscription', () => { | ||||
|       let user12 = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username12', | ||||
|             lowerCaseUsername: 'username12', | ||||
|             email: 'email12@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user12 has a 12-month recurring subscription starting today
 | ||||
|       user12.purchased.plan.customerId = 'subscribedId'; | ||||
|       user12.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user12.purchased.plan.planId = 'basic_12mo'; | ||||
|       user12.purchased.plan.consecutive.count = 0; | ||||
|       user12.purchased.plan.consecutive.offset = 12; | ||||
|       user12.purchased.plan.consecutive.trinkets = 4; | ||||
|       user12.purchased.plan.consecutive.gemCapExtra = 20; | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the first month of the first paid period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user12, tasksByType, daysMissed, analytics}); | ||||
|         expect(user12.purchased.plan.consecutive.count).to.equal(1); | ||||
|         expect(user12.purchased.plan.consecutive.offset).to.equal(11); | ||||
|         expect(user12.purchased.plan.consecutive.trinkets).to.equal(4); | ||||
|         expect(user12.purchased.plan.consecutive.gemCapExtra).to.equal(20); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the final month of the period that they already have benefits for', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(12, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user12, tasksByType, daysMissed, analytics}); | ||||
|         expect(user12.purchased.plan.consecutive.count).to.equal(12); | ||||
|         expect(user12.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user12.purchased.plan.consecutive.trinkets).to.equal(4); | ||||
|         expect(user12.purchased.plan.consecutive.gemCapExtra).to.equal(20); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the second paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(13, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user12, tasksByType, daysMissed, analytics}); | ||||
|         expect(user12.purchased.plan.consecutive.count).to.equal(13); | ||||
|         expect(user12.purchased.plan.consecutive.offset).to.equal(11); | ||||
|         expect(user12.purchased.plan.consecutive.trinkets).to.equal(8); | ||||
|         expect(user12.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits the month after the third paid period has started', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(25, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user12, tasksByType, daysMissed, analytics}); | ||||
|         expect(user12.purchased.plan.consecutive.count).to.equal(25); | ||||
|         expect(user12.purchased.plan.consecutive.offset).to.equal(11); | ||||
|         expect(user12.purchased.plan.consecutive.trinkets).to.equal(12); | ||||
|         expect(user12.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits correctly if user has been absent with continuous subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(37, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user12, tasksByType, daysMissed, analytics}); | ||||
|         expect(user12.purchased.plan.consecutive.count).to.equal(37); | ||||
|         expect(user12.purchased.plan.consecutive.offset).to.equal(11); | ||||
|         expect(user12.purchased.plan.consecutive.trinkets).to.equal(16); | ||||
|         expect(user12.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 3-month gift subscription (non-recurring)', () => { | ||||
|       let user3g = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username3g', | ||||
|             lowerCaseUsername: 'username3g', | ||||
|             email: 'email3g@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user3g has a 3-month gift subscription starting today
 | ||||
|       user3g.purchased.plan.customerId = 'Gift'; | ||||
|       user3g.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user3g.purchased.plan.dateTerminated = moment().startOf('month').add(3, 'months').add(15, 'days').toDate(); | ||||
|       user3g.purchased.plan.planId = null; | ||||
|       user3g.purchased.plan.consecutive.count = 0; | ||||
|       user3g.purchased.plan.consecutive.offset = 3; | ||||
|       user3g.purchased.plan.consecutive.trinkets = 1; | ||||
|       user3g.purchased.plan.consecutive.gemCapExtra = 5; | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the first month of the gift subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3g, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3g.purchased.plan.consecutive.count).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.offset).to.equal(2); | ||||
|         expect(user3g.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the second month of the gift subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(2, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3g, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3g.purchased.plan.consecutive.count).to.equal(2); | ||||
|         expect(user3g.purchased.plan.consecutive.offset).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the third month of the gift subscription', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(3, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3g, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3g.purchased.plan.consecutive.count).to.equal(3); | ||||
|         expect(user3g.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user3g.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.gemCapExtra).to.equal(5); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the month after the gift subscription has ended', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(4, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user3g, tasksByType, daysMissed, analytics}); | ||||
|         expect(user3g.purchased.plan.consecutive.count).to.equal(0); // subscription has been erased by now
 | ||||
|         expect(user3g.purchased.plan.consecutive.offset).to.equal(0); | ||||
|         expect(user3g.purchased.plan.consecutive.trinkets).to.equal(1); | ||||
|         expect(user3g.purchased.plan.consecutive.gemCapExtra).to.equal(0); // erased
 | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('for a 6-month recurring subscription where the user has incorrect consecutive month data from prior bugs', () => { | ||||
|       let user6x = new User({ | ||||
|         auth: { | ||||
|           local: { | ||||
|             username: 'username6x', | ||||
|             lowerCaseUsername: 'username6x', | ||||
|             email: 'email6x@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
|         }, | ||||
|       }); | ||||
|       // user6x has a 6-month recurring subscription starting 8 months in the past before issue #4819 was fixed
 | ||||
|       user6x.purchased.plan.customerId = 'subscribedId'; | ||||
|       user6x.purchased.plan.dateUpdated = moment().toDate(); | ||||
|       user6x.purchased.plan.planId = 'basic_6mo'; | ||||
|       user6x.purchased.plan.consecutive.count = 8; | ||||
|       user6x.purchased.plan.consecutive.offset = 0; | ||||
|       user6x.purchased.plan.consecutive.trinkets = 3; | ||||
|       user6x.purchased.plan.consecutive.gemCapExtra = 15; | ||||
| 
 | ||||
|       it('increments consecutive benefits in the first month since the fix for #4819 goes live', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(1, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6x, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6x.purchased.plan.consecutive.count).to.equal(9); | ||||
|         expect(user6x.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6x.purchased.plan.consecutive.trinkets).to.equal(5); | ||||
|         expect(user6x.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the second month after the fix goes live', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(2, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6x, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6x.purchased.plan.consecutive.count).to.equal(10); | ||||
|         expect(user6x.purchased.plan.consecutive.offset).to.equal(4); | ||||
|         expect(user6x.purchased.plan.consecutive.trinkets).to.equal(5); | ||||
|         expect(user6x.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not increment consecutive benefits in the third month after the fix goes live', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(3, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6x, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6x.purchased.plan.consecutive.count).to.equal(11); | ||||
|         expect(user6x.purchased.plan.consecutive.offset).to.equal(3); | ||||
|         expect(user6x.purchased.plan.consecutive.trinkets).to.equal(5); | ||||
|         expect(user6x.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
| 
 | ||||
|       it('increments consecutive benefits in the seventh month after the fix goes live', () => { | ||||
|         clock = sinon.useFakeTimers(moment().zone(0).startOf('month').add(7, 'months').add(2, 'days').toDate()); | ||||
|         cron({user: user6x, tasksByType, daysMissed, analytics}); | ||||
|         expect(user6x.purchased.plan.consecutive.count).to.equal(15); | ||||
|         expect(user6x.purchased.plan.consecutive.offset).to.equal(5); | ||||
|         expect(user6x.purchased.plan.consecutive.trinkets).to.equal(7); | ||||
|         expect(user6x.purchased.plan.consecutive.gemCapExtra).to.equal(25); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('end of the month perks when user is not subscribed', () => { | ||||
| @@ -198,14 +590,12 @@ describe('cron', () => { | ||||
|     }); | ||||
| 
 | ||||
|     it('does not reset plan.gemsBought within the month', () => { | ||||
|       let clock = sinon.useFakeTimers(moment().startOf('month').add(2, 'days').unix()); | ||||
|       clock = sinon.useFakeTimers(moment().startOf('month').add(2, 'days').unix()); | ||||
|       user.purchased.plan.dateUpdated = moment().startOf('month').toDate(); | ||||
| 
 | ||||
|       user.purchased.plan.gemsBought = 10; | ||||
|       cron({user, tasksByType, daysMissed, analytics}); | ||||
|       expect(user.purchased.plan.gemsBought).to.equal(10); | ||||
| 
 | ||||
|       clock.restore(); | ||||
|     }); | ||||
| 
 | ||||
|     it('does not reset plan.dateUpdated on a new month', () => { | ||||
| @@ -611,15 +1001,11 @@ describe('cron', () => { | ||||
| 
 | ||||
|     describe('counters', () => { | ||||
|       let notStartOfWeekOrMonth = new Date(2016, 9, 28).getTime(); // a Friday
 | ||||
|       let clock; | ||||
| 
 | ||||
|       beforeEach(() => { | ||||
|         // Replace system clocks so we can get predictable results
 | ||||
|         clock = sinon.useFakeTimers(notStartOfWeekOrMonth); | ||||
|       }); | ||||
|       afterEach(() => { | ||||
|         return clock.restore(); | ||||
|       }); | ||||
| 
 | ||||
|       it('should reset a daily habit counter each day', () => { | ||||
|         tasksByType.habits[0].counterUp = 1; | ||||
| @@ -1348,7 +1734,7 @@ describe('recoverCron', () => { | ||||
|           local: { | ||||
|             username: 'username', | ||||
|             lowerCaseUsername: 'username', | ||||
|             email: 'email@email.email', | ||||
|             email: 'email@example.com', | ||||
|             salt: 'salt', | ||||
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | ||||
|           }, | ||||
| @@ -3,9 +3,9 @@ import got from 'got'; | ||||
| import nconf from 'nconf'; | ||||
| import nodemailer from 'nodemailer'; | ||||
| import requireAgain from 'require-again'; | ||||
| import logger from '../../../../../website/server/libs/logger'; | ||||
| import { TAVERN_ID } from '../../../../../website/server/models/group'; | ||||
| import { defer } from '../../../../helpers/api-unit.helper'; | ||||
| import logger from '../../../../website/server/libs/logger'; | ||||
| import { TAVERN_ID } from '../../../../website/server/models/group'; | ||||
| import { defer } from '../../../helpers/api-unit.helper'; | ||||
| 
 | ||||
| function getUser () { | ||||
|   return { | ||||
| @@ -19,7 +19,6 @@ function getUser () { | ||||
|         emails: [{ | ||||
|           value: 'email@facebook', | ||||
|         }], | ||||
|         displayName: 'fb display name', | ||||
|       }, | ||||
|     }, | ||||
|     profile: { | ||||
| @@ -34,7 +33,7 @@ function getUser () { | ||||
| } | ||||
| 
 | ||||
| describe('emails', () => { | ||||
|   let pathToEmailLib = '../../../../../website/server/libs/email'; | ||||
|   let pathToEmailLib = '../../../../website/server/libs/email'; | ||||
| 
 | ||||
|   describe('sendEmail', () => { | ||||
|     let sendMailSpy; | ||||
| @@ -100,7 +99,7 @@ describe('emails', () => { | ||||
| 
 | ||||
|       let data = getUserInfo(user, ['name', 'email', '_id', 'canSend']); | ||||
| 
 | ||||
|       expect(data).to.have.property('name', user.auth.facebook.displayName); | ||||
|       expect(data).to.have.property('name', user.profile.name); | ||||
|       expect(data).to.have.property('email', user.auth.facebook.emails[0].value); | ||||
|       expect(data).to.have.property('_id', user._id); | ||||
|       expect(data).to.have.property('canSend', true); | ||||
| @@ -110,13 +109,12 @@ describe('emails', () => { | ||||
|       let attachEmail = requireAgain(pathToEmailLib); | ||||
|       let getUserInfo = attachEmail.getUserInfo; | ||||
|       let user = getUser(); | ||||
|       delete user.profile.name; | ||||
|       delete user.auth.local.email; | ||||
|       delete user.auth.facebook; | ||||
| 
 | ||||
|       let data = getUserInfo(user, ['name', 'email', '_id', 'canSend']); | ||||
| 
 | ||||
|       expect(data).to.have.property('name', user.auth.local.username); | ||||
|       expect(data).to.have.property('name', user.profile.name); | ||||
|       expect(data).not.to.have.property('email'); | ||||
|       expect(data).to.have.property('_id', user._id); | ||||
|       expect(data).to.have.property('canSend', true); | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   encrypt, | ||||
|   decrypt, | ||||
| } from '../../../../../website/server/libs/encryption'; | ||||
| } from '../../../../website/server/libs/encryption'; | ||||
| 
 | ||||
| describe('encryption', () => { | ||||
|   it('can encrypt and decrypt', () => { | ||||
| @@ -5,7 +5,7 @@ import { | ||||
|   BadRequest, | ||||
|   InternalServerError, | ||||
|   NotFound, | ||||
| } from '../../../../../website/server/libs/errors'; | ||||
| } from '../../../../website/server/libs/errors'; | ||||
| 
 | ||||
| describe('Custom Errors', () => { | ||||
|   describe('CustomError', () => { | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   translations, | ||||
|   localePath, | ||||
|   langCodes, | ||||
| } from '../../../../../website/server/libs/i18n'; | ||||
| } from '../../../../website/server/libs/i18n'; | ||||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
| 
 | ||||
| @@ -1,8 +1,8 @@ | ||||
| import winston from 'winston'; | ||||
| import logger from '../../../../../website/server/libs/logger'; | ||||
| import logger from '../../../../website/server/libs/logger'; | ||||
| import { | ||||
|   NotFound, | ||||
| } from '../../../../../website/server/libs//errors'; | ||||
| } from '../../../../website/server/libs//errors'; | ||||
| 
 | ||||
| describe('logger', () => { | ||||
|   let logSpy; | ||||
| @@ -2,11 +2,11 @@ | ||||
| 
 | ||||
| import { | ||||
|   encrypt, | ||||
| } from '../../../../../website/server/libs/encryption'; | ||||
| } from '../../../../website/server/libs/encryption'; | ||||
| import moment from 'moment'; | ||||
| import { | ||||
|   generateUser, | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| } from '../../../helpers/api-integration/v3'; | ||||
| import { | ||||
|   sha1Encrypt as sha1EncryptPassword, | ||||
|   sha1MakeSalt, | ||||
| @@ -15,7 +15,7 @@ import { | ||||
|   compare, | ||||
|   convertToBcrypt, | ||||
|   validatePasswordResetCodeAndFindUser, | ||||
| } from '../../../../../website/server/libs/password'; | ||||
| } from '../../../../website/server/libs/password'; | ||||
| 
 | ||||
| describe('Password Utilities', () => { | ||||
|   describe('compare', () => { | ||||
| @@ -2,11 +2,11 @@ import moment from 'moment'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import amzLib from '../../../../../../../website/server/libs/amazonPayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import amzLib from '../../../../../../website/server/libs/payments/amazon'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| import { createNonLeaderGroupMember } from '../paymentHelpers'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import amzLib from '../../../../../../../website/server/libs/amazonPayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import amzLib from '../../../../../../website/server/libs/payments/amazon'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -2,12 +2,12 @@ import cc from 'coupon-code'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Coupon } from '../../../../../../../website/server/models/coupon'; | ||||
| import amzLib from '../../../../../../../website/server/libs/amazonPayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Coupon } from '../../../../../../website/server/models/coupon'; | ||||
| import amzLib from '../../../../../../website/server/libs/payments/amazon'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -2,11 +2,11 @@ import uuid from 'uuid'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../../website/server/models/group'; | ||||
| import amzLib from '../../../../../../../website/server/libs/amazonPayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../website/server/models/group'; | ||||
| import amzLib from '../../../../../../website/server/libs/payments/amazon'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| 
 | ||||
| describe('#upgradeGroupPlan', () => { | ||||
|   let spy, data, user, group, uuidString; | ||||
| @@ -1,7 +1,7 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import iapModule from '../../../../../website/server/libs/inAppPurchases'; | ||||
| import payments from '../../../../../website/server/libs/payments'; | ||||
| import applePayments from '../../../../../website/server/libs/applePayments'; | ||||
| import payments from '../../../../../website/server/libs/payments/payments'; | ||||
| import applePayments from '../../../../../website/server/libs/payments/apple'; | ||||
| import iap from '../../../../../website/server/libs/inAppPurchases'; | ||||
| import {model as User} from '../../../../../website/server/models/user'; | ||||
| import common from '../../../../../website/common'; | ||||
| @@ -57,6 +57,18 @@ describe('Apple Payments', ()  => { | ||||
|         }); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw an error if getPurchaseData is invalid', async () => { | ||||
|       iapGetPurchaseDataStub.restore(); | ||||
|       iapGetPurchaseDataStub = sinon.stub(iapModule, 'getPurchaseData').returns([]); | ||||
| 
 | ||||
|       await expect(applePayments.verifyGemPurchase(user, receipt, headers)) | ||||
|         .to.eventually.be.rejected.and.to.eql({ | ||||
|           httpCode: 401, | ||||
|           name: 'NotAuthorized', | ||||
|           message: applePayments.constants.RESPONSE_NO_ITEM_PURCHASED, | ||||
|         }); | ||||
|     }); | ||||
| 
 | ||||
|     it('errors if the user cannot purchase gems', async () => { | ||||
|       sinon.stub(user, 'canGetGems').returnsPromise().resolves(false); | ||||
|       await expect(applePayments.verifyGemPurchase(user, receipt, headers)) | ||||
| @@ -69,27 +81,76 @@ describe('Apple Payments', ()  => { | ||||
|       user.canGetGems.restore(); | ||||
|     }); | ||||
| 
 | ||||
|     it('purchases gems', async () => { | ||||
|     it('errors if amount does not exist', async () => { | ||||
|       sinon.stub(user, 'canGetGems').returnsPromise().resolves(true); | ||||
|       await applePayments.verifyGemPurchase(user, receipt, headers); | ||||
|       iapGetPurchaseDataStub.restore(); | ||||
|       iapGetPurchaseDataStub = sinon.stub(iapModule, 'getPurchaseData') | ||||
|         .returns([{productId: 'badProduct', | ||||
|                    transactionId: token, | ||||
|         }]); | ||||
| 
 | ||||
|       expect(iapSetupStub).to.be.calledOnce; | ||||
|       expect(iapValidateStub).to.be.calledOnce; | ||||
|       expect(iapValidateStub).to.be.calledWith(iap.APPLE, receipt); | ||||
|       expect(iapIsValidatedStub).to.be.calledOnce; | ||||
|       expect(iapIsValidatedStub).to.be.calledWith({}); | ||||
|       expect(iapGetPurchaseDataStub).to.be.calledOnce; | ||||
|       await expect(applePayments.verifyGemPurchase(user, receipt, headers)) | ||||
|         .to.eventually.be.rejected.and.to.eql({ | ||||
|           httpCode: 401, | ||||
|           name: 'NotAuthorized', | ||||
|           message: applePayments.constants.RESPONSE_INVALID_ITEM, | ||||
|         }); | ||||
| 
 | ||||
|       expect(paymentBuyGemsStub).to.be.calledOnce; | ||||
|       expect(paymentBuyGemsStub).to.be.calledWith({ | ||||
|         user, | ||||
|         paymentMethod: applePayments.constants.PAYMENT_METHOD_APPLE, | ||||
|         amount: 5.25, | ||||
|         headers, | ||||
|       }); | ||||
|       expect(user.canGetGems).to.be.calledOnce; | ||||
|       user.canGetGems.restore(); | ||||
|     }); | ||||
| 
 | ||||
|     const gemsCanPurchase = [ | ||||
|       { | ||||
|         productId: 'com.habitrpg.ios.Habitica.4gems', | ||||
|         amount: 1, | ||||
|       }, | ||||
|       { | ||||
|         productId: 'com.habitrpg.ios.Habitica.20gems', | ||||
|         amount: 5.25, | ||||
|       }, | ||||
|       { | ||||
|         productId: 'com.habitrpg.ios.Habitica.21gems', | ||||
|         amount: 5.25, | ||||
|       }, | ||||
|       { | ||||
|         productId: 'com.habitrpg.ios.Habitica.42gems', | ||||
|         amount: 10.5, | ||||
|       }, | ||||
|       { | ||||
|         productId: 'com.habitrpg.ios.Habitica.84gems', | ||||
|         amount: 21, | ||||
|       }, | ||||
|     ]; | ||||
| 
 | ||||
|     gemsCanPurchase.forEach(gemTest => { | ||||
|       it(`purchases ${gemTest.productId} gems`, async () => { | ||||
|         iapGetPurchaseDataStub.restore(); | ||||
|         iapGetPurchaseDataStub = sinon.stub(iapModule, 'getPurchaseData') | ||||
|           .returns([{productId: gemTest.productId, | ||||
|                      transactionId: token, | ||||
|           }]); | ||||
| 
 | ||||
|         sinon.stub(user, 'canGetGems').returnsPromise().resolves(true); | ||||
|         await applePayments.verifyGemPurchase(user, receipt, headers); | ||||
| 
 | ||||
|         expect(iapSetupStub).to.be.calledOnce; | ||||
|         expect(iapValidateStub).to.be.calledOnce; | ||||
|         expect(iapValidateStub).to.be.calledWith(iap.APPLE, receipt); | ||||
|         expect(iapIsValidatedStub).to.be.calledOnce; | ||||
|         expect(iapIsValidatedStub).to.be.calledWith({}); | ||||
|         expect(iapGetPurchaseDataStub).to.be.calledOnce; | ||||
| 
 | ||||
|         expect(paymentBuyGemsStub).to.be.calledOnce; | ||||
|         expect(paymentBuyGemsStub).to.be.calledWith({ | ||||
|           user, | ||||
|           paymentMethod: applePayments.constants.PAYMENT_METHOD_APPLE, | ||||
|           amount: gemTest.amount, | ||||
|           headers, | ||||
|         }); | ||||
|         expect(user.canGetGems).to.be.calledOnce; | ||||
|         user.canGetGems.restore(); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('subscribe', () => { | ||||
| @@ -133,7 +194,16 @@ describe('Apple Payments', ()  => { | ||||
|       iapModule.validate.restore(); | ||||
|       iapModule.isValidated.restore(); | ||||
|       iapModule.getPurchaseData.restore(); | ||||
|       payments.createSubscription.restore(); | ||||
|       if (payments.createSubscription.restore) payments.createSubscription.restore(); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw an error if sku is empty', async () => { | ||||
|       await expect(applePayments.subscribe('', user, receipt, headers, nextPaymentProcessing)) | ||||
|         .to.eventually.be.rejected.and.to.eql({ | ||||
|           httpCode: 400, | ||||
|           name: 'BadRequest', | ||||
|           message: i18n.t('missingSubscriptionCode'), | ||||
|         }); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw an error if receipt is invalid', async () => { | ||||
| @@ -149,26 +219,69 @@ describe('Apple Payments', ()  => { | ||||
|         }); | ||||
|     }); | ||||
| 
 | ||||
|     it('creates a user subscription', async () => { | ||||
|     const subOptions = [ | ||||
|       { | ||||
|         sku: 'subscription1month', | ||||
|         subKey: 'basic_earned', | ||||
|       }, | ||||
|       { | ||||
|         sku: 'com.habitrpg.ios.habitica.subscription.3month', | ||||
|         subKey: 'basic_3mo', | ||||
|       }, | ||||
|       { | ||||
|         sku: 'com.habitrpg.ios.habitica.subscription.6month', | ||||
|         subKey: 'basic_6mo', | ||||
|       }, | ||||
|       { | ||||
|         sku: 'com.habitrpg.ios.habitica.subscription.12month', | ||||
|         subKey: 'basic_12mo', | ||||
|       }, | ||||
|     ]; | ||||
|     subOptions.forEach(option => { | ||||
|       it(`creates a user subscription for ${option.sku}`, async () => { | ||||
|         iapModule.getPurchaseData.restore(); | ||||
|         iapGetPurchaseDataStub = sinon.stub(iapModule, 'getPurchaseData') | ||||
|           .returns([{ | ||||
|             expirationDate: moment.utc().add({day: 1}).toDate(), | ||||
|             productId: option.sku, | ||||
|             transactionId: token, | ||||
|           }]); | ||||
|         sub = common.content.subscriptionBlocks[option.subKey]; | ||||
| 
 | ||||
|         await applePayments.subscribe(option.sku, user, receipt, headers, nextPaymentProcessing); | ||||
| 
 | ||||
|         expect(iapSetupStub).to.be.calledOnce; | ||||
|         expect(iapValidateStub).to.be.calledOnce; | ||||
|         expect(iapValidateStub).to.be.calledWith(iap.APPLE, receipt); | ||||
|         expect(iapIsValidatedStub).to.be.calledOnce; | ||||
|         expect(iapIsValidatedStub).to.be.calledWith({}); | ||||
|         expect(iapGetPurchaseDataStub).to.be.calledOnce; | ||||
| 
 | ||||
|         expect(paymentsCreateSubscritionStub).to.be.calledOnce; | ||||
|         expect(paymentsCreateSubscritionStub).to.be.calledWith({ | ||||
|           user, | ||||
|           customerId: token, | ||||
|           paymentMethod: applePayments.constants.PAYMENT_METHOD_APPLE, | ||||
|           sub, | ||||
|           headers, | ||||
|           additionalData: receipt, | ||||
|           nextPaymentProcessing, | ||||
|         }); | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('errors when a user is already subscribed', async () => { | ||||
|       payments.createSubscription.restore(); | ||||
|       user = new User(); | ||||
| 
 | ||||
|       await applePayments.subscribe(sku, user, receipt, headers, nextPaymentProcessing); | ||||
| 
 | ||||
|       expect(iapSetupStub).to.be.calledOnce; | ||||
|       expect(iapValidateStub).to.be.calledOnce; | ||||
|       expect(iapValidateStub).to.be.calledWith(iap.APPLE, receipt); | ||||
|       expect(iapIsValidatedStub).to.be.calledOnce; | ||||
|       expect(iapIsValidatedStub).to.be.calledWith({}); | ||||
|       expect(iapGetPurchaseDataStub).to.be.calledOnce; | ||||
| 
 | ||||
|       expect(paymentsCreateSubscritionStub).to.be.calledOnce; | ||||
|       expect(paymentsCreateSubscritionStub).to.be.calledWith({ | ||||
|         user, | ||||
|         customerId: token, | ||||
|         paymentMethod: applePayments.constants.PAYMENT_METHOD_APPLE, | ||||
|         sub, | ||||
|         headers, | ||||
|         additionalData: receipt, | ||||
|         nextPaymentProcessing, | ||||
|       }); | ||||
|       await expect(applePayments.subscribe(sku, user, receipt, headers, nextPaymentProcessing)) | ||||
|         .to.eventually.be.rejected.and.to.eql({ | ||||
|           httpCode: 401, | ||||
|           name: 'NotAuthorized', | ||||
|           message: applePayments.constants.RESPONSE_ALREADY_USED, | ||||
|         }); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
| @@ -1,7 +1,7 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import iapModule from '../../../../../website/server/libs/inAppPurchases'; | ||||
| import payments from '../../../../../website/server/libs/payments'; | ||||
| import googlePayments from '../../../../../website/server/libs/googlePayments'; | ||||
| import payments from '../../../../../website/server/libs/payments/payments'; | ||||
| import googlePayments from '../../../../../website/server/libs/payments/google'; | ||||
| import iap from '../../../../../website/server/libs/inAppPurchases'; | ||||
| import {model as User} from '../../../../../website/server/models/user'; | ||||
| import common from '../../../../../website/common'; | ||||
| @@ -1,13 +1,13 @@ | ||||
| import moment from 'moment'; | ||||
| 
 | ||||
| import * as sender from '../../../../../../../website/server/libs/email'; | ||||
| import * as api from '../../../../../../../website/server/libs/payments'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../../website/server/models/group'; | ||||
| import * as sender from '../../../../../../website/server/libs/email'; | ||||
| import * as api from '../../../../../../website/server/libs/payments/payments'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../website/server/models/group'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import i18n from '../../../../../../../website/common/script/i18n'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import i18n from '../../../../../../website/common/script/i18n'; | ||||
| 
 | ||||
| describe('Canceling a subscription for group', () => { | ||||
|   let plan, group, user, data; | ||||
| @@ -2,16 +2,16 @@ import moment from 'moment'; | ||||
| import stripeModule from 'stripe'; | ||||
| import nconf from 'nconf'; | ||||
| 
 | ||||
| import * as sender from '../../../../../../../website/server/libs/email'; | ||||
| import * as api from '../../../../../../../website/server/libs/payments'; | ||||
| import amzLib from '../../../../../../../website/server/libs/amazonPayments'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../../website/server/models/group'; | ||||
| import * as sender from '../../../../../../website/server/libs/email'; | ||||
| import * as api from '../../../../../../website/server/libs/payments/payments'; | ||||
| import amzLib from '../../../../../../website/server/libs/payments/amazon'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../website/server/models/group'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| 
 | ||||
| describe('Purchasing a group plan for group', () => { | ||||
|   const EMAIL_TEMPLATE_SUBSCRIPTION_TYPE_GOOGLE = 'Google_subscription'; | ||||
| @@ -443,8 +443,7 @@ describe('Purchasing a group plan for group', () => { | ||||
| 
 | ||||
|     await api.createSubscription(data); | ||||
| 
 | ||||
|     let updatedUser = await User.findById(recipient._id).exec(); | ||||
| 
 | ||||
|     const updatedUser = await User.findById(recipient._id).exec(); | ||||
|     expect(updatedUser.purchased.plan.extraMonths).to.within(2, 3); | ||||
|   }); | ||||
| 
 | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| 
 | ||||
| export async function createNonLeaderGroupMember (group) { | ||||
|   let nonLeader = new User(); | ||||
| @@ -1,11 +1,11 @@ | ||||
| import moment from 'moment'; | ||||
| 
 | ||||
| import * as sender from '../../../../../website/server/libs/email'; | ||||
| import * as api from '../../../../../website/server/libs/payments'; | ||||
| import * as api from '../../../../../website/server/libs/payments/payments'; | ||||
| import analytics from '../../../../../website/server/libs/analyticsService'; | ||||
| import notifications from '../../../../../website/server/libs/pushNotifications'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import { translate as t } from '../../../../helpers/api-v3-integration.helper'; | ||||
| import { translate as t } from '../../../../helpers/api-integration/v3'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../helpers/api-unit.helper.js'; | ||||
| @@ -210,7 +210,7 @@ describe('payments/index', () => { | ||||
|         let msg = '\`Hello recipient, sender has sent you 3 months of subscription!\`'; | ||||
| 
 | ||||
|         expect(user.sendMessage).to.be.calledOnce; | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: msg, senderMsg: msg }); | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: msg, senderMsg: msg, save: false }); | ||||
|       }); | ||||
| 
 | ||||
|       it('sends an email about the gift', async () => { | ||||
| @@ -629,7 +629,16 @@ describe('payments/index', () => { | ||||
|         await api.buyGems(data); | ||||
|         let msg = '\`Hello recipient, sender has sent you 4 gems!\`'; | ||||
| 
 | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: msg, senderMsg: msg }); | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: msg, senderMsg: msg, save: false }); | ||||
|       }); | ||||
| 
 | ||||
|       it('sends a message from purchaser to recipient wtih custom message', async () => { | ||||
|         data.gift.message = 'giftmessage'; | ||||
| 
 | ||||
|         await api.buyGems(data); | ||||
| 
 | ||||
|         const msg = `\`Hello recipient, sender has sent you 4 gems!\` ${data.gift.message}`; | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: msg, senderMsg: msg, save: false }); | ||||
|       }); | ||||
| 
 | ||||
|       it('sends a push notification if user did not gift to self', async () => { | ||||
| @@ -658,7 +667,7 @@ describe('payments/index', () => { | ||||
|           return `\`${messageContent}\``; | ||||
|         }); | ||||
| 
 | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: recipientsMessageContent, senderMsg: sendersMessageContent }); | ||||
|         expect(user.sendMessage).to.be.calledWith(recipient, { receiverMsg: recipientsMessageContent, senderMsg: sendersMessageContent, save: false }); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| @@ -1,7 +1,7 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| 
 | ||||
| describe('checkout success', () => { | ||||
|   const subKey = 'basic_3mo'; | ||||
| @@ -1,9 +1,9 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import nconf from 'nconf'; | ||||
| 
 | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const BASE_URL = nconf.get('BASE_URL'); | ||||
| const i18n = common.i18n; | ||||
| @@ -1,10 +1,10 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| 
 | ||||
| describe('ipn', () => { | ||||
|   const subKey = 'basic_3mo'; | ||||
| @@ -1,11 +1,11 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../website/common'; | ||||
| import { createNonLeaderGroupMember } from '../paymentHelpers'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| @@ -1,11 +1,11 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| describe('subscribeSuccess', () => { | ||||
|   const subKey = 'basic_3mo'; | ||||
| @@ -2,9 +2,9 @@ | ||||
| import moment from 'moment'; | ||||
| import cc from 'coupon-code'; | ||||
| 
 | ||||
| import paypalPayments from '../../../../../../../website/server/libs/paypalPayments'; | ||||
| import { model as Coupon } from '../../../../../../../website/server/models/coupon'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| import paypalPayments from '../../../../../../website/server/libs/payments/paypal'; | ||||
| import { model as Coupon } from '../../../../../../website/server/models/coupon'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -2,11 +2,11 @@ import stripeModule from 'stripe'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -3,12 +3,12 @@ import cc from 'coupon-code'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Coupon } from '../../../../../../../website/server/models/coupon'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Coupon } from '../../../../../../website/server/models/coupon'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -1,9 +1,9 @@ | ||||
| import stripeModule from 'stripe'; | ||||
| 
 | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -37,6 +37,22 @@ describe('checkout', () => { | ||||
|     payments.createSubscription.restore(); | ||||
|   }); | ||||
| 
 | ||||
|   it('should error if there is no token', async () => { | ||||
|     await expect(stripePayments.checkout({ | ||||
|       user, | ||||
|       gift, | ||||
|       groupId, | ||||
|       email, | ||||
|       headers, | ||||
|       coupon, | ||||
|     }, stripe)) | ||||
|       .to.eventually.be.rejected.and.to.eql({ | ||||
|         httpCode: 400, | ||||
|         message: 'Missing req.body.id', | ||||
|         name: 'BadRequest', | ||||
|       }); | ||||
|   }); | ||||
| 
 | ||||
|   it('should error if gem amount is too low', async () => { | ||||
|     let receivingUser = new User(); | ||||
|     receivingUser.save(); | ||||
| @@ -64,7 +80,6 @@ describe('checkout', () => { | ||||
|       }); | ||||
|   }); | ||||
| 
 | ||||
| 
 | ||||
|   it('should error if user cannot get gems', async () => { | ||||
|     gift = undefined; | ||||
|     sinon.stub(user, 'canGetGems').returnsPromise().resolves(false); | ||||
| @@ -2,10 +2,10 @@ import stripeModule from 'stripe'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import common from '../../../../../../website/common'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -2,12 +2,12 @@ import stripeModule from 'stripe'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| import common from '../../../../../../../website/common'; | ||||
| import logger from '../../../../../../../website/server/libs/logger'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| import common from '../../../../../../website/common'; | ||||
| import logger from '../../../../../../website/server/libs/logger'; | ||||
| import { v4 as uuid } from 'uuid'; | ||||
| import moment from 'moment'; | ||||
| 
 | ||||
| @@ -2,11 +2,11 @@ import stripeModule from 'stripe'; | ||||
| 
 | ||||
| import { | ||||
|   generateGroup, | ||||
| } from '../../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../../website/server/models/group'; | ||||
| import stripePayments from '../../../../../../../website/server/libs/stripePayments'; | ||||
| import payments from '../../../../../../../website/server/libs/payments'; | ||||
| } from '../../../../../helpers/api-unit.helper.js'; | ||||
| import { model as User } from '../../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../../website/server/models/group'; | ||||
| import stripePayments from '../../../../../../website/server/libs/payments/stripe'; | ||||
| import payments from '../../../../../../website/server/libs/payments/payments'; | ||||
| 
 | ||||
| describe('Stripe - Upgrade Group Plan', () => { | ||||
|   const stripe = stripeModule('test'); | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { preenHistory } from '../../../../../website/server/libs/preening'; | ||||
| import { preenHistory } from '../../../../website/server/libs/preening'; | ||||
| import moment from 'moment'; | ||||
| import sinon from 'sinon'; // eslint-disable-line no-shadow
 | ||||
| import { generateHistory } from '../../../../helpers/api-unit.helper.js'; | ||||
| import { generateHistory } from '../../../helpers/api-unit.helper.js'; | ||||
| 
 | ||||
| describe('preenHistory', () => { | ||||
|   let clock; | ||||
| @@ -1,13 +1,13 @@ | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import requireAgain from 'require-again'; | ||||
| import pushNotify from 'push-notify'; | ||||
| import apn from 'apn/mock'; | ||||
| import nconf from 'nconf'; | ||||
| import gcmLib from 'node-gcm'; // works with FCM notifications too
 | ||||
| 
 | ||||
| describe('pushNotifications', () => { | ||||
|   let user; | ||||
|   let sendPushNotification; | ||||
|   let pathToPushNotifications = '../../../../../website/server/libs/pushNotifications'; | ||||
|   let pathToPushNotifications = '../../../../website/server/libs/pushNotifications'; | ||||
|   let fcmSendSpy; | ||||
|   let apnSendSpy; | ||||
| 
 | ||||
| @@ -24,7 +24,7 @@ describe('pushNotifications', () => { | ||||
| 
 | ||||
|     sandbox.stub(gcmLib.Sender.prototype, 'send').callsFake(fcmSendSpy); | ||||
| 
 | ||||
|     sandbox.stub(pushNotify, 'apn').returns({ | ||||
|     sandbox.stub(apn.Provider.prototype, 'send').returns({ | ||||
|       on: () => null, | ||||
|       send: apnSendSpy, | ||||
|     }); | ||||
| @@ -104,10 +104,7 @@ describe('pushNotifications', () => { | ||||
|       }, | ||||
|     }; | ||||
| 
 | ||||
|     sendPushNotification(user, details); | ||||
|     expect(apnSendSpy).to.have.been.calledOnce; | ||||
|     expect(apnSendSpy).to.have.been.calledWithMatch({ | ||||
|       token: '123', | ||||
|     const expectedNotification = new apn.Notification({ | ||||
|       alert: message, | ||||
|       sound: 'default', | ||||
|       category: 'fun', | ||||
| @@ -117,6 +114,10 @@ describe('pushNotifications', () => { | ||||
|         b: true, | ||||
|       }, | ||||
|     }); | ||||
| 
 | ||||
|     sendPushNotification(user, details); | ||||
|     expect(apnSendSpy).to.have.been.calledOnce; | ||||
|     expect(apnSendSpy).to.have.been.calledWithMatch(expectedNotification, '123'); | ||||
|     expect(fcmSendSpy).to.not.have.been.called; | ||||
|   }); | ||||
| }); | ||||
| @@ -1,4 +1,4 @@ | ||||
| import setupNconf from '../../../../../website/server/libs/setupNconf'; | ||||
| import setupNconf from '../../../../website/server/libs/setupNconf'; | ||||
| 
 | ||||
| import path from 'path'; | ||||
| import nconf from 'nconf'; | ||||
| @@ -1,10 +1,11 @@ | ||||
| /* eslint-disable camelcase */ | ||||
| import { IncomingWebhook } from '@slack/client'; | ||||
| import requireAgain from 'require-again'; | ||||
| import slack from '../../../../../website/server/libs/slack'; | ||||
| import logger from '../../../../../website/server/libs/logger'; | ||||
| import { TAVERN_ID } from '../../../../../website/server/models/group'; | ||||
| import slack from '../../../../website/server/libs/slack'; | ||||
| import logger from '../../../../website/server/libs/logger'; | ||||
| import { TAVERN_ID } from '../../../../website/server/models/group'; | ||||
| import nconf from 'nconf'; | ||||
| import moment from 'moment'; | ||||
| 
 | ||||
| describe('slack', () => { | ||||
|   describe('sendFlagNotification', () => { | ||||
| @@ -45,13 +46,15 @@ describe('slack', () => { | ||||
|     it('sends a slack webhook', () => { | ||||
|       slack.sendFlagNotification(data); | ||||
| 
 | ||||
|       const timestamp = `${moment(data.message.timestamp).utc().format('YYYY-MM-DD HH:mm')} UTC`; | ||||
| 
 | ||||
|       expect(IncomingWebhook.prototype.send).to.be.calledOnce; | ||||
|       expect(IncomingWebhook.prototype.send).to.be.calledWith({ | ||||
|         text: 'flagger (flagger-id; language: flagger-lang) flagged a message', | ||||
|         attachments: [{ | ||||
|           fallback: 'Flag Message', | ||||
|           color: 'danger', | ||||
|           author_name: 'Author - author@example.com - author-id', | ||||
|           author_name: `Author - author@example.com - author-id\n${timestamp}`, | ||||
|           title: 'Flag in Some group - (private guild)', | ||||
|           title_link: undefined, | ||||
|           text: 'some text', | ||||
| @@ -97,9 +100,11 @@ describe('slack', () => { | ||||
| 
 | ||||
|       slack.sendFlagNotification(data); | ||||
| 
 | ||||
|       const timestamp = `${moment(data.message.timestamp).utc().format('YYYY-MM-DD HH:mm')} UTC`; | ||||
| 
 | ||||
|       expect(IncomingWebhook.prototype.send).to.be.calledWithMatch({ | ||||
|         attachments: [sandbox.match({ | ||||
|           author_name: 'System Message', | ||||
|           author_name: `System Message\n${timestamp}`, | ||||
|         })], | ||||
|       }); | ||||
|     }); | ||||
| @@ -107,7 +112,7 @@ describe('slack', () => { | ||||
|     it('noops if no flagging url is provided', () => { | ||||
|       sandbox.stub(nconf, 'get').withArgs('SLACK:FLAGGING_URL').returns(''); | ||||
|       sandbox.stub(logger, 'error'); | ||||
|       let reRequiredSlack = requireAgain('../../../../../website/server/libs/slack'); | ||||
|       let reRequiredSlack = requireAgain('../../../../website/server/libs/slack'); | ||||
| 
 | ||||
|       expect(logger.error).to.be.calledOnce; | ||||
| 
 | ||||
| @@ -3,13 +3,13 @@ import { | ||||
|   getTasks, | ||||
|   syncableAttrs, | ||||
|   moveTask, | ||||
| } from '../../../../../website/server/libs/taskManager'; | ||||
| import i18n from '../../../../../website/common/script/i18n'; | ||||
| } from '../../../../website/server/libs/taskManager'; | ||||
| import i18n from '../../../../website/common/script/i18n'; | ||||
| import { | ||||
|   generateUser, | ||||
|   generateGroup, | ||||
|   generateChallenge, | ||||
| } from '../../../../helpers/api-unit.helper.js'; | ||||
| } from '../../../helpers/api-unit.helper.js'; | ||||
| 
 | ||||
| describe('taskManager', () => { | ||||
|   let user, group, challenge; | ||||
| @@ -4,11 +4,19 @@ import { | ||||
|   taskScoredWebhook, | ||||
|   groupChatReceivedWebhook, | ||||
|   taskActivityWebhook, | ||||
| } from '../../../../../website/server/libs/webhook'; | ||||
| import { defer } from '../../../../helpers/api-unit.helper'; | ||||
|   questActivityWebhook, | ||||
|   userActivityWebhook, | ||||
| } from '../../../../website/server/libs/webhook'; | ||||
| import { | ||||
|   model as User, | ||||
| } from '../../../../website/server/models/user'; | ||||
| import { | ||||
|   generateUser, | ||||
| } from '../../../helpers/api-unit.helper.js'; | ||||
| import { defer } from '../../../helpers/api-unit.helper'; | ||||
| 
 | ||||
| describe('webhooks', () => { | ||||
|   let webhooks; | ||||
|   let webhooks, user; | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     sandbox.stub(got, 'post').returns(defer().promise); | ||||
| @@ -23,6 +31,26 @@ describe('webhooks', () => { | ||||
|         updated: true, | ||||
|         deleted: true, | ||||
|         scored: true, | ||||
|         checklistScored: true, | ||||
|       }, | ||||
|     }, { | ||||
|       id: 'questActivity', | ||||
|       url: 'http://quest-activity.com', | ||||
|       enabled: true, | ||||
|       type: 'questActivity', | ||||
|       options: { | ||||
|         questStarted: true, | ||||
|         questFinised: true, | ||||
|       }, | ||||
|     }, { | ||||
|       id: 'userActivity', | ||||
|       url: 'http://user-activity.com', | ||||
|       enabled: true, | ||||
|       type: 'userActivity', | ||||
|       options: { | ||||
|         petHatched: true, | ||||
|         mountRaised: true, | ||||
|         leveledUp: true, | ||||
|       }, | ||||
|     }, { | ||||
|       id: 'groupChatReceived', | ||||
| @@ -33,6 +61,9 @@ describe('webhooks', () => { | ||||
|         groupId: 'group-id', | ||||
|       }, | ||||
|     }]; | ||||
| 
 | ||||
|     user = generateUser(); | ||||
|     user.webhooks = webhooks; | ||||
|   }); | ||||
| 
 | ||||
|   afterEach(() => { | ||||
| @@ -57,7 +88,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(WebhookSender.defaultTransformData).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledOnce; | ||||
| @@ -67,6 +99,30 @@ describe('webhooks', () => { | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('adds default data (user and webhookType) to the body', () => { | ||||
|       let sendWebhook = new WebhookSender({ | ||||
|         type: 'custom', | ||||
|       }); | ||||
|       sandbox.spy(sendWebhook, 'attachDefaultData'); | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(sendWebhook.attachDefaultData).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch('http://custom-url.com', { | ||||
|         json: true, | ||||
|       }); | ||||
| 
 | ||||
|       expect(body).to.eql({ | ||||
|         foo: 'bar', | ||||
|         user: {_id: user._id}, | ||||
|         webhookType: 'custom', | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('can pass in a data transformation function', () => { | ||||
|       sandbox.spy(WebhookSender, 'defaultTransformData'); | ||||
|       let sendWebhook = new WebhookSender({ | ||||
| @@ -80,7 +136,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(WebhookSender.defaultTransformData).to.not.be.called; | ||||
|       expect(got.post).to.be.calledOnce; | ||||
| @@ -93,7 +150,7 @@ describe('webhooks', () => { | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('provieds a default filter function', () => { | ||||
|     it('provides a default filter function', () => { | ||||
|       sandbox.spy(WebhookSender, 'defaultWebhookFilter'); | ||||
|       let sendWebhook = new WebhookSender({ | ||||
|         type: 'custom', | ||||
| @@ -101,7 +158,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(WebhookSender.defaultWebhookFilter).to.be.calledOnce; | ||||
|     }); | ||||
| @@ -117,7 +175,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(WebhookSender.defaultWebhookFilter).to.not.be.called; | ||||
|       expect(got.post).to.not.be.called; | ||||
| @@ -134,10 +193,11 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([ | ||||
|       user.webhooks = [ | ||||
|         { id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom', options: { foo: 'bar' }}, | ||||
|         { id: 'other-custom-webhook', url: 'http://other-custom-url.com', enabled: true, type: 'custom', options: { foo: 'foo' }}, | ||||
|       ], body); | ||||
|       ]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch('http://custom-url.com'); | ||||
| @@ -150,7 +210,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'http://custom-url.com', enabled: false, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'http://custom-url.com', enabled: false, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.not.be.called; | ||||
|     }); | ||||
| @@ -162,7 +223,8 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([{id: 'custom-webhook', url: 'httxp://custom-url!!', enabled: true, type: 'custom'}], body); | ||||
|       user.webhooks = [{id: 'custom-webhook', url: 'httxp://custom-url!!!', enabled: true, type: 'custom'}]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.not.be.called; | ||||
|     }); | ||||
| @@ -174,10 +236,30 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([ | ||||
|       user.webhooks = [ | ||||
|         { id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}, | ||||
|         { id: 'other-webhook', url: 'http://other-url.com', enabled: true, type: 'other'}, | ||||
|       ], body); | ||||
|       ]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch('http://custom-url.com', { | ||||
|         body, | ||||
|         json: true, | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('sends every type of activity to global webhooks', () => { | ||||
|       let sendWebhook = new WebhookSender({ | ||||
|         type: 'custom', | ||||
|       }); | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       user.webhooks = [ | ||||
|         { id: 'global-webhook', url: 'http://custom-url.com', enabled: true, type: 'globalActivity'}, | ||||
|       ]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch('http://custom-url.com', { | ||||
| @@ -193,10 +275,11 @@ describe('webhooks', () => { | ||||
| 
 | ||||
|       let body = { foo: 'bar' }; | ||||
| 
 | ||||
|       sendWebhook.send([ | ||||
|       user.webhooks = [ | ||||
|         { id: 'custom-webhook', url: 'http://custom-url.com', enabled: true, type: 'custom'}, | ||||
|         { id: 'other-custom-webhook', url: 'http://other-url.com', enabled: true, type: 'custom'}, | ||||
|       ], body); | ||||
|       ]; | ||||
|       sendWebhook.send(user, body); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledTwice; | ||||
|       expect(got.post).to.be.calledWithMatch('http://custom-url.com', { | ||||
| @@ -216,7 +299,6 @@ describe('webhooks', () => { | ||||
|     beforeEach(() => { | ||||
|       data = { | ||||
|         user: { | ||||
|           _id: 'user-id', | ||||
|           _tmp: {foo: 'bar'}, | ||||
|           stats: { | ||||
|             lvl: 5, | ||||
| @@ -227,17 +309,6 @@ describe('webhooks', () => { | ||||
|               return this; | ||||
|             }, | ||||
|           }, | ||||
|           addComputedStatsToJSONObj () { | ||||
|             let mockStats = Object.assign({ | ||||
|               maxHealth: 50, | ||||
|               maxMP: 103, | ||||
|               toNextLevel: 40, | ||||
|             }, this.stats); | ||||
| 
 | ||||
|             delete mockStats.toJSON; | ||||
| 
 | ||||
|             return mockStats; | ||||
|           }, | ||||
|         }, | ||||
|         task: { | ||||
|           text: 'text', | ||||
| @@ -245,18 +316,66 @@ describe('webhooks', () => { | ||||
|         direction: 'up', | ||||
|         delta: 176, | ||||
|       }; | ||||
| 
 | ||||
|       let mockStats = Object.assign({ | ||||
|         maxHealth: 50, | ||||
|         maxMP: 103, | ||||
|         toNextLevel: 40, | ||||
|       }, data.user.stats); | ||||
|       delete mockStats.toJSON; | ||||
| 
 | ||||
|       sandbox.stub(User, 'addComputedStatsToJSONObj').returns(mockStats); | ||||
|     }); | ||||
| 
 | ||||
|     it('sends task and stats data', () => { | ||||
|       taskScoredWebhook.send(webhooks, data); | ||||
|       taskScoredWebhook.send(user, data); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch(webhooks[0].url, { | ||||
|         json: true, | ||||
|         body: { | ||||
|           type: 'scored', | ||||
|           webhookType: 'taskActivity', | ||||
|           user: { | ||||
|             _id: 'user-id', | ||||
|             _id: user._id, | ||||
|             _tmp: {foo: 'bar'}, | ||||
|             stats: { | ||||
|               lvl: 5, | ||||
|               int: 10, | ||||
|               str: 5, | ||||
|               exp: 423, | ||||
|               toNextLevel: 40, | ||||
|               maxHealth: 50, | ||||
|               maxMP: 103, | ||||
|             }, | ||||
|           }, | ||||
|           task: { | ||||
|             text: 'text', | ||||
|           }, | ||||
|           direction: 'up', | ||||
|           delta: 176, | ||||
|         }, | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     it('sends task and stats data to globalActivity webhookd', () => { | ||||
|       user.webhooks = [{ | ||||
|         id: 'globalActivity', | ||||
|         url: 'http://global-activity.com', | ||||
|         enabled: true, | ||||
|         type: 'globalActivity', | ||||
|       }]; | ||||
| 
 | ||||
|       taskScoredWebhook.send(user, data); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch('http://global-activity.com', { | ||||
|         json: true, | ||||
|         body: { | ||||
|           type: 'scored', | ||||
|           webhookType: 'taskActivity', | ||||
|           user: { | ||||
|             _id: user._id, | ||||
|             _tmp: {foo: 'bar'}, | ||||
|             stats: { | ||||
|               lvl: 5, | ||||
| @@ -280,7 +399,7 @@ describe('webhooks', () => { | ||||
|     it('does not send task scored data if scored option is not true', () => { | ||||
|       webhooks[0].options.scored = false; | ||||
| 
 | ||||
|       taskScoredWebhook.send(webhooks, data); | ||||
|       taskScoredWebhook.send(user, data); | ||||
| 
 | ||||
|       expect(got.post).to.not.be.called; | ||||
|     }); | ||||
| @@ -301,13 +420,17 @@ describe('webhooks', () => { | ||||
|       it(`sends ${type} tasks`, () => { | ||||
|         data.type = type; | ||||
| 
 | ||||
|         taskActivityWebhook.send(webhooks, data); | ||||
|         taskActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.be.calledOnce; | ||||
|         expect(got.post).to.be.calledWithMatch(webhooks[0].url, { | ||||
|           json: true, | ||||
|           body: { | ||||
|             type, | ||||
|             webhookType: 'taskActivity', | ||||
|             user: { | ||||
|               _id: user._id, | ||||
|             }, | ||||
|             task: data.task, | ||||
|           }, | ||||
|         }); | ||||
| @@ -317,7 +440,142 @@ describe('webhooks', () => { | ||||
|         data.type = type; | ||||
|         webhooks[0].options[type] = false; | ||||
| 
 | ||||
|         taskActivityWebhook.send(webhooks, data); | ||||
|         taskActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.not.be.called; | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
|     describe('checklistScored', () => { | ||||
|       beforeEach(() => { | ||||
|         data = { | ||||
|           task: { | ||||
|             text: 'text', | ||||
|           }, | ||||
|           item: { | ||||
|             text: 'item-text', | ||||
|           }, | ||||
|         }; | ||||
|       }); | ||||
| 
 | ||||
|       it('sends \'checklistScored\' tasks', () => { | ||||
|         data.type = 'checklistScored'; | ||||
| 
 | ||||
|         taskActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.be.calledOnce; | ||||
|         expect(got.post).to.be.calledWithMatch(webhooks[0].url, { | ||||
|           json: true, | ||||
|           body: { | ||||
|             webhookType: 'taskActivity', | ||||
|             user: { | ||||
|               _id: user._id, | ||||
|             }, | ||||
|             type: data.type, | ||||
|             task: data.task, | ||||
|             item: data.item, | ||||
|           }, | ||||
|         }); | ||||
|       }); | ||||
| 
 | ||||
|       it('does not send task \'checklistScored\' data if \'checklistScored\' option is not true', () => { | ||||
|         data.type = 'checklistScored'; | ||||
|         webhooks[0].options.checklistScored = false; | ||||
| 
 | ||||
|         taskActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.not.be.called; | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('userActivityWebhook', () => { | ||||
|     let data; | ||||
| 
 | ||||
|     beforeEach(() => { | ||||
|       data = { | ||||
|         something: true, | ||||
|       }; | ||||
|     }); | ||||
| 
 | ||||
|     ['petHatched', 'mountRaised', 'leveledUp'].forEach((type) => { | ||||
|       it(`sends ${type} webhooks`, () => { | ||||
|         data.type = type; | ||||
| 
 | ||||
|         userActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.be.calledOnce; | ||||
|         expect(got.post).to.be.calledWithMatch(webhooks[2].url, { | ||||
|           json: true, | ||||
|           body: { | ||||
|             type, | ||||
|             webhookType: 'userActivity', | ||||
|             user: { | ||||
|               _id: user._id, | ||||
|             }, | ||||
|             something: true, | ||||
|           }, | ||||
|         }); | ||||
|       }); | ||||
| 
 | ||||
|       it(`does not send webhook ${type} data if ${type} option is not true`, () => { | ||||
|         data.type = type; | ||||
|         webhooks[2].options[type] = false; | ||||
| 
 | ||||
|         userActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.not.be.called; | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('questActivityWebhook', () => { | ||||
|     let data; | ||||
| 
 | ||||
|     beforeEach(() => { | ||||
|       data = { | ||||
|         group: { | ||||
|           id: 'group-id', | ||||
|           name: 'some group', | ||||
|           otherData: 'foo', | ||||
|         }, | ||||
|         quest: { | ||||
|           key: 'some-key', | ||||
|         }, | ||||
|       }; | ||||
|     }); | ||||
| 
 | ||||
|     ['questStarted', 'questFinised'].forEach((type) => { | ||||
|       it(`sends ${type} webhooks`, () => { | ||||
|         data.type = type; | ||||
| 
 | ||||
|         questActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.be.calledOnce; | ||||
|         expect(got.post).to.be.calledWithMatch(webhooks[1].url, { | ||||
|           json: true, | ||||
|           body: { | ||||
|             type, | ||||
|             webhookType: 'questActivity', | ||||
|             user: { | ||||
|               _id: user._id, | ||||
|             }, | ||||
|             group: { | ||||
|               id: 'group-id', | ||||
|               name: 'some group', | ||||
|             }, | ||||
|             quest: { | ||||
|               key: 'some-key', | ||||
|             }, | ||||
|           }, | ||||
|         }); | ||||
|       }); | ||||
| 
 | ||||
|       it(`does not send webhook ${type} data if ${type} option is not true`, () => { | ||||
|         data.type = type; | ||||
|         webhooks[1].options[type] = false; | ||||
| 
 | ||||
|         userActivityWebhook.send(user, data); | ||||
| 
 | ||||
|         expect(got.post).to.not.be.called; | ||||
|       }); | ||||
| @@ -338,12 +596,16 @@ describe('webhooks', () => { | ||||
|         }, | ||||
|       }; | ||||
| 
 | ||||
|       groupChatReceivedWebhook.send(webhooks, data); | ||||
|       groupChatReceivedWebhook.send(user, data); | ||||
| 
 | ||||
|       expect(got.post).to.be.calledOnce; | ||||
|       expect(got.post).to.be.calledWithMatch(webhooks[webhooks.length - 1].url, { | ||||
|         json: true, | ||||
|         body: { | ||||
|           webhookType: 'groupChatReceived', | ||||
|           user: { | ||||
|             _id: user._id, | ||||
|           }, | ||||
|           group: { | ||||
|             id: 'group-id', | ||||
|             name: 'some group', | ||||
| @@ -369,7 +631,7 @@ describe('webhooks', () => { | ||||
|         }, | ||||
|       }; | ||||
| 
 | ||||
|       groupChatReceivedWebhook.send(webhooks, data); | ||||
|       groupChatReceivedWebhook.send(user, data); | ||||
| 
 | ||||
|       expect(got.post).to.not.be.called; | ||||
|     }); | ||||
| @@ -3,14 +3,14 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import analyticsService from '../../../../../website/server/libs/analyticsService'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import analyticsService from '../../../../website/server/libs/analyticsService'; | ||||
| import nconf from 'nconf'; | ||||
| import requireAgain from 'require-again'; | ||||
| 
 | ||||
| describe('analytics middleware', () => { | ||||
|   let res, req, next; | ||||
|   let pathToAnalyticsMiddleware = '../../../../../website/server/middlewares/analytics'; | ||||
|   let pathToAnalyticsMiddleware = '../../../../website/server/middlewares/analytics'; | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     res = generateRes(); | ||||
							
								
								
									
										40
									
								
								test/api/unit/middlewares/auth.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								test/api/unit/middlewares/auth.test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import { authWithHeaders as authWithHeadersFactory } from '../../../../website/server/middlewares/auth'; | ||||
|  | ||||
| describe('auth middleware', () => { | ||||
|   let res, req, user; | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     res = generateRes(); | ||||
|     req = generateReq(); | ||||
|     user = await res.locals.user.save(); | ||||
|   }); | ||||
|  | ||||
|   describe('auth with headers', () => { | ||||
|     it('allows to specify a list of user field that we do not want to load', (done) => { | ||||
|       const authWithHeaders = authWithHeadersFactory({ | ||||
|         userFieldsToExclude: ['items', 'flags', 'auth.timestamps'], | ||||
|       }); | ||||
|  | ||||
|       req.headers['x-api-user'] = user._id; | ||||
|       req.headers['x-api-key'] = user.apiToken; | ||||
|  | ||||
|       authWithHeaders(req, res, (err) => { | ||||
|         if (err) return done(err); | ||||
|  | ||||
|         const userToJSON = res.locals.user.toJSON(); | ||||
|         expect(userToJSON.items).to.not.exist; | ||||
|         expect(userToJSON.flags).to.not.exist; | ||||
|         expect(userToJSON.auth.timestamps).to.not.exist; | ||||
|         expect(userToJSON.auth).to.exist; | ||||
|         expect(userToJSON.notifications).to.exist; | ||||
|         expect(userToJSON.preferences).to.exist; | ||||
|  | ||||
|         done(); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
| @@ -3,8 +3,8 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import cors from '../../../../../website/server/middlewares/cors'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import cors from '../../../../website/server/middlewares/cors'; | ||||
| 
 | ||||
| describe('cors middleware', () => { | ||||
|   let res, req, next; | ||||
| @@ -3,14 +3,14 @@ import { | ||||
|   generateReq, | ||||
|   generateTodo, | ||||
|   generateDaily, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import cronMiddleware from '../../../../../website/server/middlewares/cron'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import cronMiddleware from '../../../../website/server/middlewares/cron'; | ||||
| import moment from 'moment'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../website/server/models/group'; | ||||
| import * as Tasks from '../../../../../website/server/models/task'; | ||||
| import analyticsService from '../../../../../website/server/libs/analyticsService'; | ||||
| import * as cronLib from '../../../../../website/server/libs/cron'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../website/server/models/group'; | ||||
| import * as Tasks from '../../../../website/server/models/task'; | ||||
| import analyticsService from '../../../../website/server/libs/analyticsService'; | ||||
| import * as cronLib from '../../../../website/server/libs/cron'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
| 
 | ||||
| const CRON_TIMEOUT_WAIT = new Date(60 * 60 * 1000).getTime(); | ||||
| @@ -166,8 +166,11 @@ describe('cron middleware', () => { | ||||
|     await new Promise((resolve, reject) => { | ||||
|       cronMiddleware(req, res, (err) => { | ||||
|         if (err) return reject(err); | ||||
|         expect(user.stats.hp).to.be.lessThan(hpBefore); | ||||
|         resolve(); | ||||
|         User.findOne({_id: user._id}, function (secondErr, updatedUser) { | ||||
|           if (secondErr) return reject(secondErr); | ||||
|           expect(updatedUser.stats.hp).to.be.lessThan(hpBefore); | ||||
|           resolve(); | ||||
|         }); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| @@ -176,7 +179,7 @@ describe('cron middleware', () => { | ||||
|     user.lastCron = moment(new Date()).subtract({days: 2}); | ||||
|     let todo = generateTodo(user); | ||||
|     let todoValueBefore = todo.value; | ||||
|     await user.save(); | ||||
|     await Promise.all([todo.save(), user.save()]); | ||||
| 
 | ||||
|     await new Promise((resolve, reject) => { | ||||
|       cronMiddleware(req, res, (err) => { | ||||
| @@ -217,8 +220,11 @@ describe('cron middleware', () => { | ||||
|     await new Promise((resolve, reject) => { | ||||
|       cronMiddleware(req, res, (err) => { | ||||
|         if (err) return reject(err); | ||||
|         expect(user.stats.hp).to.be.lessThan(hpBefore); | ||||
|         resolve(); | ||||
|         User.findOne({_id: user._id}, function (secondErr, updatedUser) { | ||||
|           if (secondErr) return reject(secondErr); | ||||
|           expect(updatedUser.stats.hp).to.be.lessThan(hpBefore); | ||||
|           resolve(); | ||||
|         }); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| @@ -3,11 +3,11 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import i18n from '../../../../../website/common/script/i18n'; | ||||
| import { ensureAdmin, ensureSudo } from '../../../../../website/server/middlewares/ensureAccessRight'; | ||||
| import { NotAuthorized } from '../../../../../website/server/libs/errors'; | ||||
| import apiMessages from '../../../../../website/server/libs/apiMessages'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import i18n from '../../../../website/common/script/i18n'; | ||||
| import { ensureAdmin, ensureSudo } from '../../../../website/server/middlewares/ensureAccessRight'; | ||||
| import { NotAuthorized } from '../../../../website/server/libs/errors'; | ||||
| import apiError from '../../../../website/server/libs/apiError'; | ||||
| 
 | ||||
| describe('ensure access middlewares', () => { | ||||
|   let res, req, next; | ||||
| @@ -46,7 +46,7 @@ describe('ensure access middlewares', () => { | ||||
|       ensureSudo(req, res, next); | ||||
| 
 | ||||
|       const calledWith = next.getCall(0).args; | ||||
|       expect(calledWith[0].message).to.equal(apiMessages('noSudoAccess')); | ||||
|       expect(calledWith[0].message).to.equal(apiError('noSudoAccess')); | ||||
|       expect(calledWith[0] instanceof NotAuthorized).to.equal(true); | ||||
|     }); | ||||
| 
 | ||||
| @@ -3,9 +3,9 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import ensureDevelpmentMode from '../../../../../website/server/middlewares/ensureDevelpmentMode'; | ||||
| import { NotFound } from '../../../../../website/server/libs/errors'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import ensureDevelpmentMode from '../../../../website/server/middlewares/ensureDevelpmentMode'; | ||||
| import { NotFound } from '../../../../website/server/libs/errors'; | ||||
| import nconf from 'nconf'; | ||||
| 
 | ||||
| describe('developmentMode middleware', () => { | ||||
| @@ -2,17 +2,17 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| 
 | ||||
| import errorHandler from '../../../../../website/server/middlewares/errorHandler'; | ||||
| import responseMiddleware from '../../../../../website/server/middlewares/response'; | ||||
| import errorHandler from '../../../../website/server/middlewares/errorHandler'; | ||||
| import responseMiddleware from '../../../../website/server/middlewares/response'; | ||||
| import { | ||||
|   getUserLanguage, | ||||
|   attachTranslateFunction, | ||||
| } from '../../../../../website/server/middlewares/language'; | ||||
| } from '../../../../website/server/middlewares/language'; | ||||
| 
 | ||||
| import { BadRequest } from '../../../../../website/server/libs/errors'; | ||||
| import logger from '../../../../../website/server/libs/logger'; | ||||
| import { BadRequest } from '../../../../website/server/libs/errors'; | ||||
| import logger from '../../../../website/server/libs/logger'; | ||||
| 
 | ||||
| describe('errorHandler', () => { | ||||
|   let res, req, next; | ||||
| @@ -2,13 +2,13 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import { | ||||
|   getUserLanguage, | ||||
|   attachTranslateFunction, | ||||
| } from '../../../../../website/server/middlewares/language'; | ||||
| import common from '../../../../../website/common'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| } from '../../../../website/server/middlewares/language'; | ||||
| import common from '../../../../website/common'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| 
 | ||||
| const i18n = common.i18n; | ||||
| 
 | ||||
| @@ -2,13 +2,13 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import nconf from 'nconf'; | ||||
| import requireAgain from 'require-again'; | ||||
| 
 | ||||
| describe('maintenance mode middleware', () => { | ||||
|   let res, req, next; | ||||
|   let pathToMaintenanceModeMiddleware = '../../../../../website/server/middlewares/maintenanceMode'; | ||||
|   let pathToMaintenanceModeMiddleware = '../../../../website/server/middlewares/maintenanceMode'; | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     res = generateRes(); | ||||
| @@ -2,13 +2,13 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import nconf from 'nconf'; | ||||
| import requireAgain from 'require-again'; | ||||
| 
 | ||||
| describe('redirects middleware', () => { | ||||
|   let res, req, next; | ||||
|   let pathToRedirectsMiddleware = '../../../../../website/server/middlewares/redirects'; | ||||
|   let pathToRedirectsMiddleware = '../../../../website/server/middlewares/redirects'; | ||||
| 
 | ||||
|   beforeEach(() => { | ||||
|     res = generateRes(); | ||||
| @@ -2,9 +2,9 @@ import { | ||||
|   generateRes, | ||||
|   generateReq, | ||||
|   generateNext, | ||||
| } from '../../../../helpers/api-unit.helper'; | ||||
| import responseMiddleware from '../../../../../website/server/middlewares/response'; | ||||
| import packageInfo from '../../../../../package.json'; | ||||
| } from '../../../helpers/api-unit.helper'; | ||||
| import responseMiddleware from '../../../../website/server/middlewares/response'; | ||||
| import packageInfo from '../../../../package.json'; | ||||
| 
 | ||||
| describe('response middleware', () => { | ||||
|   let res, req, next; | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { model as Challenge } from '../../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../../website/server/models/task'; | ||||
| import common from '../../../../../website/common/'; | ||||
| import { model as Challenge } from '../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../website/server/models/task'; | ||||
| import common from '../../../../website/common/'; | ||||
| import { each, find } from 'lodash'; | ||||
| 
 | ||||
| describe('Challenge Model', () => { | ||||
| @@ -1,26 +1,30 @@ | ||||
| import moment from 'moment'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
| import validator from 'validator'; | ||||
| import { sleep } from '../../../../helpers/api-unit.helper'; | ||||
| import { sleep } from '../../../helpers/api-unit.helper'; | ||||
| import { | ||||
|   SPAM_MESSAGE_LIMIT, | ||||
|   SPAM_MIN_EXEMPT_CONTRIB_LEVEL, | ||||
|   SPAM_WINDOW_LENGTH, | ||||
|   INVITES_LIMIT, | ||||
|   model as Group, | ||||
| } from '../../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import { quests as questScrolls } from '../../../../../website/common/script/content'; | ||||
| import { groupChatReceivedWebhook } from '../../../../../website/server/libs/webhook'; | ||||
| import * as email from '../../../../../website/server/libs/email'; | ||||
| import { TAVERN_ID } from '../../../../../website/common/script/'; | ||||
| import shared from '../../../../../website/common'; | ||||
| } from '../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import { quests as questScrolls } from '../../../../website/common/script/content'; | ||||
| import { | ||||
|   groupChatReceivedWebhook, | ||||
|   questActivityWebhook, | ||||
| } from '../../../../website/server/libs/webhook'; | ||||
| import * as email from '../../../../website/server/libs/email'; | ||||
| import { TAVERN_ID } from '../../../../website/common/script/'; | ||||
| import shared from '../../../../website/common'; | ||||
| 
 | ||||
| describe('Group Model', () => { | ||||
|   let party, questLeader, participatingMember, nonParticipatingMember, undecidedMember; | ||||
| 
 | ||||
|   beforeEach(async () => { | ||||
|     sandbox.stub(email, 'sendTxn'); | ||||
|     sandbox.stub(questActivityWebhook, 'send'); | ||||
| 
 | ||||
|     party = new Group({ | ||||
|       name: 'test party', | ||||
| @@ -182,7 +186,7 @@ describe('Group Model', () => { | ||||
|           await party.startQuest(questLeader); | ||||
|           await party.save(); | ||||
| 
 | ||||
|           sendChatStub = sandbox.stub(Group.prototype, 'sendChat'); | ||||
|           sendChatStub = sandbox.spy(Group.prototype, 'sendChat'); | ||||
|         }); | ||||
| 
 | ||||
|         afterEach(() => sendChatStub.restore()); | ||||
| @@ -378,7 +382,7 @@ describe('Group Model', () => { | ||||
|           await party.startQuest(questLeader); | ||||
|           await party.save(); | ||||
| 
 | ||||
|           sendChatStub = sandbox.stub(Group.prototype, 'sendChat'); | ||||
|           sendChatStub = sandbox.spy(Group.prototype, 'sendChat'); | ||||
|         }); | ||||
| 
 | ||||
|         afterEach(() => sendChatStub.restore()); | ||||
| @@ -918,21 +922,8 @@ describe('Group Model', () => { | ||||
|         sandbox.spy(User, 'update'); | ||||
|       }); | ||||
| 
 | ||||
|       it('puts message at top of chat array', () => { | ||||
|         let oldMessage = { | ||||
|           text: 'a message', | ||||
|         }; | ||||
|         party.chat.push(oldMessage, oldMessage, oldMessage); | ||||
| 
 | ||||
|         party.sendChat('a new message', {_id: 'user-id', profile: { name: 'user name' }}); | ||||
| 
 | ||||
|         expect(party.chat).to.have.a.lengthOf(4); | ||||
|         expect(party.chat[0].text).to.eql('a new message'); | ||||
|         expect(party.chat[0].uuid).to.eql('user-id'); | ||||
|       }); | ||||
| 
 | ||||
|       it('formats message', () => { | ||||
|         party.sendChat('a new message', { | ||||
|         const chatMessage = party.sendChat('a new message', { | ||||
|           _id: 'user-id', | ||||
|           profile: { name: 'user name' }, | ||||
|           contributor: { | ||||
| @@ -947,11 +938,11 @@ describe('Group Model', () => { | ||||
|           }, | ||||
|         }); | ||||
| 
 | ||||
|         let chat = party.chat[0]; | ||||
|         const chat = chatMessage; | ||||
| 
 | ||||
|         expect(chat.text).to.eql('a new message'); | ||||
|         expect(validator.isUUID(chat.id)).to.eql(true); | ||||
|         expect(chat.timestamp).to.be.a('number'); | ||||
|         expect(chat.timestamp).to.be.a('date'); | ||||
|         expect(chat.likes).to.eql({}); | ||||
|         expect(chat.flags).to.eql({}); | ||||
|         expect(chat.flagCount).to.eql(0); | ||||
| @@ -962,13 +953,11 @@ describe('Group Model', () => { | ||||
|       }); | ||||
| 
 | ||||
|       it('formats message as system if no user is passed in', () => { | ||||
|         party.sendChat('a system message'); | ||||
| 
 | ||||
|         let chat = party.chat[0]; | ||||
|         const chat = party.sendChat('a system message'); | ||||
| 
 | ||||
|         expect(chat.text).to.eql('a system message'); | ||||
|         expect(validator.isUUID(chat.id)).to.eql(true); | ||||
|         expect(chat.timestamp).to.be.a('number'); | ||||
|         expect(chat.timestamp).to.be.a('date'); | ||||
|         expect(chat.likes).to.eql({}); | ||||
|         expect(chat.flags).to.eql({}); | ||||
|         expect(chat.flagCount).to.eql(0); | ||||
| @@ -1204,6 +1193,47 @@ describe('Group Model', () => { | ||||
|           expect(typeOfEmail).to.eql('quest-started'); | ||||
|         }); | ||||
| 
 | ||||
|         it('sends webhook to participating members that quest has started', async () => { | ||||
|           // should receive webhook
 | ||||
|           participatingMember.webhooks = [{ | ||||
|             type: 'questActivity', | ||||
|             url: 'http://someurl.com', | ||||
|             options: { | ||||
|               questStarted: true, | ||||
|             }, | ||||
|           }]; | ||||
|           questLeader.webhooks = [{ | ||||
|             type: 'questActivity', | ||||
|             url: 'http://someurl.com', | ||||
|             options: { | ||||
|               questStarted: true, | ||||
|             }, | ||||
|           }]; | ||||
| 
 | ||||
|           await Promise.all([participatingMember.save(), questLeader.save()]); | ||||
| 
 | ||||
|           await party.startQuest(nonParticipatingMember); | ||||
| 
 | ||||
|           await sleep(0.5); | ||||
| 
 | ||||
|           expect(questActivityWebhook.send).to.be.calledTwice; // for 2 participating members
 | ||||
| 
 | ||||
|           let args = questActivityWebhook.send.args[0]; | ||||
|           let webhooks = args[0].webhooks; | ||||
|           let webhookOwner = args[0]._id; | ||||
|           let options = args[1]; | ||||
| 
 | ||||
|           expect(webhooks).to.have.a.lengthOf(1); | ||||
|           if (webhookOwner === questLeader._id) { | ||||
|             expect(webhooks[0].id).to.eql(questLeader.webhooks[0].id); | ||||
|           } else { | ||||
|             expect(webhooks[0].id).to.eql(participatingMember.webhooks[0].id); | ||||
|           } | ||||
|           expect(webhooks[0].type).to.eql('questActivity'); | ||||
|           expect(options.group).to.eql(party); | ||||
|           expect(options.quest.key).to.eql('whale'); | ||||
|         }); | ||||
| 
 | ||||
|         it('sends email only to members who have not opted out', async () => { | ||||
|           participatingMember.preferences.emailNotifications.questStarted = false; | ||||
|           questLeader.preferences.emailNotifications.questStarted = true; | ||||
| @@ -1375,7 +1405,8 @@ describe('Group Model', () => { | ||||
|         expect(updatedParticipatingMember.achievements.quests[quest.key]).to.eql(1); | ||||
|       }); | ||||
| 
 | ||||
|       it('gives out super awesome Masterclasser achievement to the deserving', async () => { | ||||
|       // Disable test, it fails on TravisCI, but only there
 | ||||
|       xit('gives out super awesome Masterclasser achievement to the deserving', async () => { | ||||
|         quest = questScrolls.lostMasterclasser4; | ||||
|         party.quest.key = quest.key; | ||||
| 
 | ||||
| @@ -1411,7 +1442,8 @@ describe('Group Model', () => { | ||||
|         expect(updatedParticipatingMember.achievements.lostMasterclasser).to.not.eql(true); | ||||
|       }); | ||||
| 
 | ||||
|       it('gives out super awesome Masterclasser achievement when quests done out of order', async () => { | ||||
|       // Disable test, it fails on TravisCI, but only there
 | ||||
|       xit('gives out super awesome Masterclasser achievement when quests done out of order', async () => { | ||||
|         quest = questScrolls.lostMasterclasser1; | ||||
|         party.quest.key = quest.key; | ||||
| 
 | ||||
| @@ -1583,6 +1615,42 @@ describe('Group Model', () => { | ||||
|         }); | ||||
|       }); | ||||
| 
 | ||||
|       it('sends webhook to participating members that quest has finished', async () => { | ||||
|         // should receive webhook
 | ||||
|         participatingMember.webhooks = [{ | ||||
|           type: 'questActivity', | ||||
|           url: 'http://someurl.com', | ||||
|           options: { | ||||
|             questFinished: true, | ||||
|           }, | ||||
|         }]; | ||||
|         questLeader.webhooks = [{ | ||||
|           type: 'questActivity', | ||||
|           url: 'http://someurl.com', | ||||
|           options: { | ||||
|             questStarted: true, // will not receive the webhook
 | ||||
|           }, | ||||
|         }]; | ||||
| 
 | ||||
|         await Promise.all([participatingMember.save(), questLeader.save()]); | ||||
| 
 | ||||
|         await party.finishQuest(quest); | ||||
| 
 | ||||
|         await sleep(0.5); | ||||
| 
 | ||||
|         expect(questActivityWebhook.send).to.be.calledOnce; | ||||
| 
 | ||||
|         let args = questActivityWebhook.send.args[0]; | ||||
|         let webhooks = args[0].webhooks; | ||||
|         let options = args[1]; | ||||
| 
 | ||||
|         expect(webhooks).to.have.a.lengthOf(1); | ||||
|         expect(webhooks[0].id).to.eql(participatingMember.webhooks[0].id); | ||||
|         expect(webhooks[0].type).to.eql('questActivity'); | ||||
|         expect(options.group).to.eql(party); | ||||
|         expect(options.quest.key).to.eql(quest.key); | ||||
|       }); | ||||
| 
 | ||||
|       context('World quests in Tavern', () => { | ||||
|         let tavernQuest; | ||||
| 
 | ||||
| @@ -1698,7 +1766,7 @@ describe('Group Model', () => { | ||||
|         expect(groupChatReceivedWebhook.send).to.be.calledOnce; | ||||
| 
 | ||||
|         let args = groupChatReceivedWebhook.send.args[0]; | ||||
|         let webhooks = args[0]; | ||||
|         let webhooks = args[0].webhooks; | ||||
|         let options = args[1]; | ||||
| 
 | ||||
|         expect(webhooks).to.have.a.lengthOf(1); | ||||
| @@ -1762,9 +1830,9 @@ describe('Group Model', () => { | ||||
|         expect(groupChatReceivedWebhook.send).to.be.calledThrice; | ||||
| 
 | ||||
|         let args = groupChatReceivedWebhook.send.args; | ||||
|         expect(args.find(arg => arg[0][0].id === memberWithWebhook.webhooks[0].id)).to.be.exist; | ||||
|         expect(args.find(arg => arg[0][0].id === memberWithWebhook2.webhooks[0].id)).to.be.exist; | ||||
|         expect(args.find(arg => arg[0][0].id === memberWithWebhook3.webhooks[0].id)).to.be.exist; | ||||
|         expect(args.find(arg => arg[0].webhooks[0].id === memberWithWebhook.webhooks[0].id)).to.be.exist; | ||||
|         expect(args.find(arg => arg[0].webhooks[0].id === memberWithWebhook2.webhooks[0].id)).to.be.exist; | ||||
|         expect(args.find(arg => arg[0].webhooks[0].id === memberWithWebhook3.webhooks[0].id)).to.be.exist; | ||||
|       }); | ||||
|     }); | ||||
| 
 | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { model as Challenge } from '../../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../../website/server/models/task'; | ||||
| import { model as Challenge } from '../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../website/server/models/task'; | ||||
| import { each, find, findIndex } from 'lodash'; | ||||
| 
 | ||||
| describe('Group Task Methods', () => { | ||||
| @@ -1,10 +1,10 @@ | ||||
| import { model as Challenge } from '../../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../../website/server/models/task'; | ||||
| import { InternalServerError } from '../../../../../website/server/libs/errors'; | ||||
| import { model as Challenge } from '../../../../website/server/models/challenge'; | ||||
| import { model as Group } from '../../../../website/server/models/group'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import * as Tasks from '../../../../website/server/models/task'; | ||||
| import { InternalServerError } from '../../../../website/server/libs/errors'; | ||||
| import { each } from 'lodash'; | ||||
| import { generateHistory } from '../../../../helpers/api-unit.helper.js'; | ||||
| import { generateHistory } from '../../../helpers/api-unit.helper.js'; | ||||
| 
 | ||||
| describe('Task Model', () => { | ||||
|   let guild, leader, challenge, task; | ||||
| @@ -1,7 +1,7 @@ | ||||
| import moment from 'moment'; | ||||
| import { model as User } from '../../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../../website/server/models/group'; | ||||
| import common from '../../../../../website/common'; | ||||
| import { model as User } from '../../../../website/server/models/user'; | ||||
| import { model as Group } from '../../../../website/server/models/group'; | ||||
| import common from '../../../../website/common'; | ||||
| 
 | ||||
| describe('User Model', () => { | ||||
|   it('keeps user._tmp when calling .toJSON', () => { | ||||
| @@ -42,13 +42,48 @@ describe('User Model', () => { | ||||
|     expect(userToJSON.stats.maxHealth).to.not.exist; | ||||
|     expect(userToJSON.stats.toNextLevel).to.not.exist; | ||||
| 
 | ||||
|     user.addComputedStatsToJSONObj(userToJSON.stats); | ||||
|     User.addComputedStatsToJSONObj(userToJSON.stats, userToJSON); | ||||
| 
 | ||||
|     expect(userToJSON.stats.maxMP).to.exist; | ||||
|     expect(userToJSON.stats.maxHealth).to.equal(common.maxHealth); | ||||
|     expect(userToJSON.stats.toNextLevel).to.equal(common.tnl(user.stats.lvl)); | ||||
|   }); | ||||
| 
 | ||||
|   it('can transform user object without mongoose helpers', async () => { | ||||
|     let user = new User(); | ||||
|     await user.save(); | ||||
|     let userToJSON = await User.findById(user._id).lean().exec(); | ||||
| 
 | ||||
|     expect(userToJSON.stats.maxMP).to.not.exist; | ||||
|     expect(userToJSON.stats.maxHealth).to.not.exist; | ||||
|     expect(userToJSON.stats.toNextLevel).to.not.exist; | ||||
|     expect(userToJSON.id).to.not.exist; | ||||
| 
 | ||||
|     User.transformJSONUser(userToJSON); | ||||
| 
 | ||||
|     expect(userToJSON.id).to.equal(userToJSON._id); | ||||
|     expect(userToJSON.stats.maxMP).to.not.exist; | ||||
|     expect(userToJSON.stats.maxHealth).to.not.exist; | ||||
|     expect(userToJSON.stats.toNextLevel).to.not.exist; | ||||
|   }); | ||||
| 
 | ||||
|   it('can transform user object without mongoose helpers (including computed stats)', async () => { | ||||
|     let user = new User(); | ||||
|     await user.save(); | ||||
|     let userToJSON = await User.findById(user._id).lean().exec(); | ||||
| 
 | ||||
|     expect(userToJSON.stats.maxMP).to.not.exist; | ||||
|     expect(userToJSON.stats.maxHealth).to.not.exist; | ||||
|     expect(userToJSON.stats.toNextLevel).to.not.exist; | ||||
| 
 | ||||
|     User.transformJSONUser(userToJSON, true); | ||||
| 
 | ||||
|     expect(userToJSON.id).to.equal(userToJSON._id); | ||||
|     expect(userToJSON.stats.maxMP).to.exist; | ||||
|     expect(userToJSON.stats.maxHealth).to.equal(common.maxHealth); | ||||
|     expect(userToJSON.stats.toNextLevel).to.equal(common.tnl(user.stats.lvl)); | ||||
|   }); | ||||
| 
 | ||||
|   context('notifications', () => { | ||||
|     it('can add notifications without data', () => { | ||||
|       let user = new User(); | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { model as UserNotification } from '../../../../../website/server/models/userNotification'; | ||||
| import { model as UserNotification } from '../../../../website/server/models/userNotification'; | ||||
| 
 | ||||
| describe('UserNotification Model', () => { | ||||
|   context('convertNotificationsToSafeJson', () => { | ||||
							
								
								
									
										323
									
								
								test/api/unit/models/webhook.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										323
									
								
								test/api/unit/models/webhook.test.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,323 @@ | ||||
| import { model as Webhook } from '../../../../website/server/models/webhook'; | ||||
| import { BadRequest } from '../../../../website/server/libs/errors'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
| import apiError from '../../../../website/server/libs/apiError'; | ||||
|  | ||||
| describe('Webhook Model', () => { | ||||
|   context('Instance Methods', () => { | ||||
|     describe('#formatOptions', () => { | ||||
|       let res; | ||||
|  | ||||
|       beforeEach(() => { | ||||
|         res = { | ||||
|           t: sandbox.spy(), | ||||
|         }; | ||||
|       }); | ||||
|       context('type is taskActivity', () => { | ||||
|         let config; | ||||
|  | ||||
|         beforeEach(() => { | ||||
|           config = { | ||||
|             type: 'taskActivity', | ||||
|             url: 'https//exmaple.com/endpoint', | ||||
|             options: { | ||||
|               created: true, | ||||
|               updated: true, | ||||
|               deleted: true, | ||||
|               scored: true, | ||||
|               checklistScored: true, | ||||
|             }, | ||||
|           }; | ||||
|         }); | ||||
|  | ||||
|         it('it provides default values for options', () => { | ||||
|           delete config.options; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             checklistScored: false, | ||||
|             created: false, | ||||
|             updated: false, | ||||
|             deleted: false, | ||||
|             scored: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('provides missing task options', () => { | ||||
|           delete config.options.created; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             checklistScored: true, | ||||
|             created: false, | ||||
|             updated: true, | ||||
|             deleted: true, | ||||
|             scored: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('discards additional options', () => { | ||||
|           config.options.foo = 'another option'; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options.foo).to.not.exist; | ||||
|           expect(wh.options).to.eql({ | ||||
|             checklistScored: true, | ||||
|             created: true, | ||||
|             updated: true, | ||||
|             deleted: true, | ||||
|             scored: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         ['created', 'updated', 'deleted', 'scored', 'checklistScored'].forEach((option) => { | ||||
|           it(`validates that ${option} is a boolean`, (done) => { | ||||
|             config.options[option] = 'not a boolean'; | ||||
|  | ||||
|             try { | ||||
|               let wh = new Webhook(config); | ||||
|  | ||||
|               wh.formatOptions(res); | ||||
|             } catch (err) { | ||||
|               expect(err).to.be.an.instanceOf(BadRequest); | ||||
|               expect(res.t).to.be.calledOnce; | ||||
|               expect(res.t).to.be.calledWith('webhookBooleanOption', { option }); | ||||
|               done(); | ||||
|             } | ||||
|           }); | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|       context('type is userActivity', () => { | ||||
|         let config; | ||||
|  | ||||
|         beforeEach(() => { | ||||
|           config = { | ||||
|             type: 'userActivity', | ||||
|             url: 'https//exmaple.com/endpoint', | ||||
|             options: { | ||||
|               petHatched: true, | ||||
|               mountRaised: true, | ||||
|               leveledUp: true, | ||||
|             }, | ||||
|           }; | ||||
|         }); | ||||
|  | ||||
|         it('it provides default values for options', () => { | ||||
|           delete config.options; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             petHatched: false, | ||||
|             mountRaised: false, | ||||
|             leveledUp: false, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('provides missing user options', () => { | ||||
|           delete config.options.petHatched; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             petHatched: false, | ||||
|             mountRaised: true, | ||||
|             leveledUp: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('discards additional options', () => { | ||||
|           config.options.foo = 'another option'; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options.foo).to.not.exist; | ||||
|           expect(wh.options).to.eql({ | ||||
|             petHatched: true, | ||||
|             mountRaised: true, | ||||
|             leveledUp: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         ['petHatched', 'petHatched', 'leveledUp'].forEach((option) => { | ||||
|           it(`validates that ${option} is a boolean`, (done) => { | ||||
|             config.options[option] = 'not a boolean'; | ||||
|  | ||||
|             try { | ||||
|               let wh = new Webhook(config); | ||||
|  | ||||
|               wh.formatOptions(res); | ||||
|             } catch (err) { | ||||
|               expect(err).to.be.an.instanceOf(BadRequest); | ||||
|               expect(res.t).to.be.calledOnce; | ||||
|               expect(res.t).to.be.calledWith('webhookBooleanOption', { option }); | ||||
|               done(); | ||||
|             } | ||||
|           }); | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|       context('type is questActivity', () => { | ||||
|         let config; | ||||
|  | ||||
|         beforeEach(() => { | ||||
|           config = { | ||||
|             type: 'questActivity', | ||||
|             url: 'https//exmaple.com/endpoint', | ||||
|             options: { | ||||
|               questStarted: true, | ||||
|               questFinished: true, | ||||
|             }, | ||||
|           }; | ||||
|         }); | ||||
|  | ||||
|         it('it provides default values for options', () => { | ||||
|           delete config.options; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             questStarted: false, | ||||
|             questFinished: false, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('provides missing user options', () => { | ||||
|           delete config.options.questStarted; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql({ | ||||
|             questStarted: false, | ||||
|             questFinished: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('discards additional options', () => { | ||||
|           config.options.foo = 'another option'; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options.foo).to.not.exist; | ||||
|           expect(wh.options).to.eql({ | ||||
|             questStarted: true, | ||||
|             questFinished: true, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         ['questStarted', 'questFinished'].forEach((option) => { | ||||
|           it(`validates that ${option} is a boolean`, (done) => { | ||||
|             config.options[option] = 'not a boolean'; | ||||
|  | ||||
|             try { | ||||
|               let wh = new Webhook(config); | ||||
|  | ||||
|               wh.formatOptions(res); | ||||
|             } catch (err) { | ||||
|               expect(err).to.be.an.instanceOf(BadRequest); | ||||
|               expect(res.t).to.be.calledOnce; | ||||
|               expect(res.t).to.be.calledWith('webhookBooleanOption', { option }); | ||||
|               done(); | ||||
|             } | ||||
|           }); | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|       context('type is groupChatReceived', () => { | ||||
|         let config; | ||||
|  | ||||
|         beforeEach(() => { | ||||
|           config = { | ||||
|             type: 'groupChatReceived', | ||||
|             url: 'https//exmaple.com/endpoint', | ||||
|             options: { | ||||
|               groupId: generateUUID(), | ||||
|             }, | ||||
|           }; | ||||
|         }); | ||||
|  | ||||
|         it('creates options', () => { | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options).to.eql(config.options); | ||||
|         }); | ||||
|  | ||||
|         it('discards additional objects', () => { | ||||
|           config.options.foo = 'another thing'; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options.foo).to.not.exist; | ||||
|           expect(wh.options).to.eql({ | ||||
|             groupId: config.options.groupId, | ||||
|           }); | ||||
|         }); | ||||
|  | ||||
|         it('requires groupId option to be a uuid', (done) => { | ||||
|           config.options.groupId = 'not a uuid'; | ||||
|  | ||||
|           try { | ||||
|             let wh = new Webhook(config); | ||||
|  | ||||
|             wh.formatOptions(res); | ||||
|           } catch (err) { | ||||
|             expect(err).to.be.an.instanceOf(BadRequest); | ||||
|  | ||||
|             expect(err.message).to.eql(apiError('groupIdRequired')); | ||||
|             done(); | ||||
|           } | ||||
|         }); | ||||
|       }); | ||||
|  | ||||
|  | ||||
|       context('type is globalActivity', () => { | ||||
|         let config; | ||||
|  | ||||
|         beforeEach(() => { | ||||
|           config = { | ||||
|             type: 'globalActivity', | ||||
|             url: 'https//exmaple.com/endpoint', | ||||
|             options: { }, | ||||
|           }; | ||||
|         }); | ||||
|  | ||||
|         it('discards additional objects', () => { | ||||
|           config.options.foo = 'another thing'; | ||||
|  | ||||
|           let wh = new Webhook(config); | ||||
|  | ||||
|           wh.formatOptions(res); | ||||
|  | ||||
|           expect(wh.options.foo).to.not.exist; | ||||
|           expect(wh.options).to.eql({}); | ||||
|         }); | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
| @@ -5,7 +5,7 @@ import { | ||||
|   sleep, | ||||
|   checkExistence, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('DELETE /challenges/:challengeId', () => { | ||||
| @@ -41,6 +41,7 @@ describe('DELETE /challenges/:challengeId', () => { | ||||
|       group = populatedGroup.group; | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|       await groupLeader.post(`/tasks/challenge/${challenge._id}`, [ | ||||
|         {type: 'habit', text: taskText}, | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   createAndPopulateGroup, | ||||
|   generateChallenge, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('GET /challenges/:challengeId', () => { | ||||
| @@ -33,9 +33,11 @@ describe('GET /challenges/:challengeId', () => { | ||||
|       group = populatedGroup.group; | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return challenge data', async () => { | ||||
|       await challenge.sync(); | ||||
|       let chal = await user.get(`/challenges/${challenge._id}`); | ||||
|       expect(chal.memberCount).to.equal(challenge.memberCount); | ||||
|       expect(chal.name).to.equal(challenge.name); | ||||
| @@ -80,6 +82,7 @@ describe('GET /challenges/:challengeId', () => { | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await members[0].post(`/challenges/${challenge._id}/join`); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('fails if user doesn\'t have access to the challenge', async () => { | ||||
| @@ -134,6 +137,7 @@ describe('GET /challenges/:challengeId', () => { | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await members[0].post(`/challenges/${challenge._id}/join`); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('fails if user doesn\'t have access to the challenge', async () => { | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import { | ||||
|   generateChallenge, | ||||
|   translate as t, | ||||
|   sleep, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('GET /challenges/:challengeId/export/csv', () => { | ||||
| @@ -24,6 +24,7 @@ describe('GET /challenges/:challengeId/export/csv', () => { | ||||
|     members = populatedGroup.members; | ||||
|  | ||||
|     challenge = await generateChallenge(groupLeader, group); | ||||
|     await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     await members[0].post(`/challenges/${challenge._id}/join`); | ||||
|     await members[1].post(`/challenges/${challenge._id}/join`); | ||||
|     await members[2].post(`/challenges/${challenge._id}/join`); | ||||
| @@ -60,9 +61,9 @@ describe('GET /challenges/:challengeId/export/csv', () => { | ||||
|   }); | ||||
|  | ||||
|   it('should return a valid CSV file with export data', async () => { | ||||
|     let res = await members[0].get(`/challenges/${challenge._id}/export/csv`); | ||||
|     let sortedMembers = _.sortBy([members[0], members[1], members[2], groupLeader], '_id'); | ||||
|     let splitRes = res.split('\n'); | ||||
|     const res = await members[0].get(`/challenges/${challenge._id}/export/csv`); | ||||
|     const sortedMembers = _.sortBy([members[0], members[1], members[2], groupLeader], '_id'); | ||||
|     const splitRes = res.split('\n'); | ||||
|  | ||||
|     expect(splitRes[0]).to.equal('UUID,name,Task,Value,Notes,Streak,Task,Value,Notes,Streak'); | ||||
|     expect(splitRes[1]).to.equal(`${sortedMembers[0]._id},${sortedMembers[0].profile.name},habit:Task 1,0,,0,todo:Task 2,0,,0`); | ||||
| @@ -71,4 +72,16 @@ describe('GET /challenges/:challengeId/export/csv', () => { | ||||
|     expect(splitRes[4]).to.equal(`${sortedMembers[3]._id},${sortedMembers[3].profile.name},habit:Task 1,0,,0,todo:Task 2,0,,0`); | ||||
|     expect(splitRes[5]).to.equal(''); | ||||
|   }); | ||||
|  | ||||
|   it('should successfully return when it contains erroneous residue user data', async () => { | ||||
|     await members[0].update({challenges: []}); | ||||
|     const res = await members[1].get(`/challenges/${challenge._id}/export/csv`); | ||||
|     const sortedMembers = _.sortBy([members[1], members[2], groupLeader], '_id'); | ||||
|     const splitRes = res.split('\n'); | ||||
|     expect(splitRes[0]).to.equal('UUID,name,Task,Value,Notes,Streak,Task,Value,Notes,Streak'); | ||||
|     expect(splitRes[1]).to.equal(`${sortedMembers[0]._id},${sortedMembers[0].profile.name},habit:Task 1,0,,0,todo:Task 2,0,,0`); | ||||
|     expect(splitRes[2]).to.equal(`${sortedMembers[1]._id},${sortedMembers[1].profile.name},habit:Task 1,0,,0,todo:Task 2,0,,0`); | ||||
|     expect(splitRes[3]).to.equal(`${sortedMembers[2]._id},${sortedMembers[2].profile.name},habit:Task 1,0,,0,todo:Task 2,0,,0`); | ||||
|     expect(splitRes[4]).to.equal(''); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateGroup, | ||||
|   generateChallenge, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('GET /challenges/:challengeId/members', () => { | ||||
| @@ -45,6 +45,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|     let leader = await generateUser({balance: 4}); | ||||
|     let group = await generateGroup(leader, {type: 'guild', privacy: 'public', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(leader, group); | ||||
|     await leader.post(`/challenges/${challenge._id}/join`); | ||||
|     let res = await user.get(`/challenges/${challenge._id}/members`); | ||||
|     expect(res[0]).to.eql({ | ||||
|       _id: leader._id, | ||||
| @@ -59,6 +60,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|     let anotherUser = await generateUser({balance: 3}); | ||||
|     let group = await generateGroup(anotherUser, {type: 'guild', privacy: 'public', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(anotherUser, group); | ||||
|     await anotherUser.post(`/challenges/${challenge._id}/join`); | ||||
|     let res = await user.get(`/challenges/${challenge._id}/members`); | ||||
|     expect(res[0]).to.eql({ | ||||
|       _id: anotherUser._id, | ||||
| @@ -72,6 +74,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|   it('returns only first 30 members if req.query.includeAllMembers is not true', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|     let usersToGenerate = []; | ||||
|     for (let i = 0; i < 31; i++) { | ||||
| @@ -90,6 +93,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|   it('returns only first 30 members if req.query.includeAllMembers is not defined', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|     let usersToGenerate = []; | ||||
|     for (let i = 0; i < 31; i++) { | ||||
| @@ -108,6 +112,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|   it('returns all members if req.query.includeAllMembers is true', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|     let usersToGenerate = []; | ||||
|     for (let i = 0; i < 31; i++) { | ||||
| @@ -123,9 +128,11 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   it('supports using req.query.lastId to get more members', async () => { | ||||
|   it('supports using req.query.lastId to get more members', async function () { | ||||
|     this.timeout(30000); // @TODO: times out after 8 seconds | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|     let usersToGenerate = []; | ||||
|     for (let i = 0; i < 57; i++) { | ||||
| @@ -146,6 +153,7 @@ describe('GET /challenges/:challengeId/members', () => { | ||||
|   it('supports using req.query.search to get search members', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|     let usersToGenerate = []; | ||||
|     for (let i = 0; i < 3; i++) { | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateChallenge, | ||||
|   generateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
| @@ -50,6 +50,7 @@ describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
|   it('fails if user doesn\'t have access to the challenge', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|     let anotherUser = await generateUser(); | ||||
|     let member = await generateUser(); | ||||
|     await expect(anotherUser.get(`/challenges/${challenge._id}/members/${member._id}`)).to.eventually.be.rejected.and.eql({ | ||||
| @@ -62,6 +63,7 @@ describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
|   it('fails if member is not part of the challenge', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|     let member = await generateUser(); | ||||
|     await expect(user.get(`/challenges/${challenge._id}/members/${member._id}`)).to.eventually.be.rejected.and.eql({ | ||||
|       code: 404, | ||||
| @@ -74,6 +76,7 @@ describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
|     let groupLeader = await generateUser({balance: 4}); | ||||
|     let group = await generateGroup(groupLeader, {type: 'guild', privacy: 'public', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(groupLeader, group); | ||||
|     await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     let taskText = 'Test Text'; | ||||
|     await groupLeader.post(`/tasks/challenge/${challenge._id}`, [{type: 'habit', text: taskText}]); | ||||
|  | ||||
| @@ -86,6 +89,7 @@ describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
|   it('returns the member tasks for the challenges', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|     await user.post(`/tasks/challenge/${challenge._id}`, [{type: 'habit', text: 'Test Text'}]); | ||||
|  | ||||
|     let memberProgress = await user.get(`/challenges/${challenge._id}/members/${user._id}`); | ||||
| @@ -98,6 +102,7 @@ describe('GET /challenges/:challengeId/members/:memberId', () => { | ||||
|   it('returns the tasks without the tags and checklist', async () => { | ||||
|     let group = await generateGroup(user, {type: 'party', name: generateUUID()}); | ||||
|     let challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|     let taskText = 'Test Text'; | ||||
|     await user.post(`/tasks/challenge/${challenge._id}`, [{ | ||||
|       type: 'todo', | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateChallenge, | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { TAVERN_ID } from '../../../../../website/common/script/constants'; | ||||
|  | ||||
| describe('GET challenges/groups/:groupId', () => { | ||||
| @@ -25,7 +25,9 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       nonMember = await generateUser(); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return group challenges for non member with populated leader', async () => { | ||||
| @@ -73,6 +75,7 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       expect(foundChallengeIndex).to.eql(0); | ||||
|  | ||||
|       let newChallenge = await generateChallenge(user, publicGuild); | ||||
|       await user.post(`/challenges/${newChallenge._id}/join`); | ||||
|  | ||||
|       challenges = await user.get(`/challenges/groups/${publicGuild._id}`); | ||||
|  | ||||
| @@ -99,7 +102,9 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       nonMember = await generateUser(); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should prevent non-member from seeing challenges', async () => { | ||||
| @@ -156,9 +161,12 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|           slug: 'habitica_official', | ||||
|         }], | ||||
|       }); | ||||
|       await user.post(`/challenges/${officialChallenge._id}/join`); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return official challenges first', async () => { | ||||
| @@ -178,6 +186,7 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       expect(foundChallengeIndex).to.eql(1); | ||||
|  | ||||
|       let newChallenge = await generateChallenge(user, publicGuild); | ||||
|       await user.post(`/challenges/${newChallenge._id}/join`); | ||||
|  | ||||
|       challenges = await user.get(`/challenges/groups/${publicGuild._id}`); | ||||
|  | ||||
| @@ -203,7 +212,9 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       nonMember = await generateUser(); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should prevent non-member from seeing challenges', async () => { | ||||
| @@ -263,7 +274,9 @@ describe('GET challenges/groups/:groupId', () => { | ||||
|       tavern = await user.get(`/groups/${TAVERN_ID}`); | ||||
|  | ||||
|       challenge = await generateChallenge(user, tavern, {prize: 1}); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, tavern, {prize: 1}); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return tavern challenges with populated leader', async () => { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   generateUser, | ||||
|   generateChallenge, | ||||
|   createAndPopulateGroup, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
|  | ||||
| describe('GET challenges/user', () => { | ||||
|   context('no official challenges', () => { | ||||
| @@ -24,7 +24,9 @@ describe('GET challenges/user', () => { | ||||
|       nonMember = await generateUser(); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return challenges user has joined', async () => { | ||||
| @@ -146,6 +148,7 @@ describe('GET challenges/user', () => { | ||||
|       expect(foundChallengeIndex).to.eql(0); | ||||
|  | ||||
|       let newChallenge = await generateChallenge(user, publicGuild); | ||||
|       await user.post(`/challenges/${newChallenge._id}/join`); | ||||
|  | ||||
|       challenges = await user.get('/challenges/user'); | ||||
|  | ||||
| @@ -164,6 +167,7 @@ describe('GET challenges/user', () => { | ||||
|       }); | ||||
|  | ||||
|       let privateChallenge = await generateChallenge(groupLeader, group); | ||||
|       await groupLeader.post(`/challenges/${privateChallenge._id}/join`); | ||||
|  | ||||
|       let challenges = await nonMember.get('/challenges/user'); | ||||
|  | ||||
| @@ -198,9 +202,12 @@ describe('GET challenges/user', () => { | ||||
|           slug: 'habitica_official', | ||||
|         }], | ||||
|       }); | ||||
|       await user.post(`/challenges/${officialChallenge._id}/join`); | ||||
|  | ||||
|       challenge = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge._id}/join`); | ||||
|       challenge2 = await generateChallenge(user, group); | ||||
|       await user.post(`/challenges/${challenge2._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('should return official challenges first', async () => { | ||||
| @@ -220,6 +227,7 @@ describe('GET challenges/user', () => { | ||||
|       expect(foundChallengeIndex).to.eql(1); | ||||
|  | ||||
|       let newChallenge = await generateChallenge(user, publicGuild); | ||||
|       await user.post(`/challenges/${newChallenge._id}/join`); | ||||
|  | ||||
|       challenges = await user.get('/challenges/user'); | ||||
|  | ||||
| @@ -252,12 +260,14 @@ describe('GET challenges/user', () => { | ||||
|       await user.update({balance: 20}); | ||||
|  | ||||
|       for (let i = 0; i < 11; i += 1) { | ||||
|         await generateChallenge(user, group); // eslint-disable-line | ||||
|         let challenge = await generateChallenge(user, group); // eslint-disable-line | ||||
|         await user.post(`/challenges/${challenge._id}/join`); // eslint-disable-line | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     it('returns public guilds filtered by category', async () => { | ||||
|       const categoryChallenge = await generateChallenge(user, guild, {categories}); | ||||
|       await user.post(`/challenges/${categoryChallenge._id}/join`); | ||||
|       const challenges = await user.get(`/challenges/user?categories=${categories[0].slug}`); | ||||
|  | ||||
|       expect(challenges[0]._id).to.eql(categoryChallenge._id); | ||||
| @@ -267,7 +277,7 @@ describe('GET challenges/user', () => { | ||||
|     it('does not page challenges if page parameter is absent', async () => { | ||||
|       const challenges = await user.get('/challenges/user'); | ||||
|  | ||||
|       expect(challenges.length).to.eql(12); | ||||
|       expect(challenges.length).to.be.above(11); | ||||
|     }); | ||||
|  | ||||
|     it('paginates challenges', async () => { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   generateUser, | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('POST /challenges', () => { | ||||
| @@ -242,7 +242,6 @@ describe('POST /challenges', () => { | ||||
|     it('returns an error when challenge validation fails; doesn\'s save user or group', async () => { | ||||
|       let oldChallengeCount = group.challengeCount; | ||||
|       let oldUserBalance = groupLeader.balance; | ||||
|       let oldUserChallenges = groupLeader.challenges; | ||||
|       let oldGroupBalance = group.balance; | ||||
|  | ||||
|       await expect(groupLeader.post('/challenges', { | ||||
| @@ -260,7 +259,6 @@ describe('POST /challenges', () => { | ||||
|       expect(group.challengeCount).to.eql(oldChallengeCount); | ||||
|       expect(group.balance).to.eql(oldGroupBalance); | ||||
|       expect(groupLeader.balance).to.eql(oldUserBalance); | ||||
|       expect(groupLeader.challenges).to.eql(oldUserChallenges); | ||||
|     }); | ||||
|  | ||||
|     it('sets all properites of the challenge as passed', async () => { | ||||
| @@ -291,18 +289,19 @@ describe('POST /challenges', () => { | ||||
|         name: group.name, | ||||
|         type: group.type, | ||||
|       }); | ||||
|       expect(challenge.memberCount).to.eql(1); | ||||
|       expect(challenge.memberCount).to.eql(0); | ||||
|       expect(challenge.prize).to.eql(prize); | ||||
|     }); | ||||
|  | ||||
|     it('adds challenge to creator\'s challenges', async () => { | ||||
|       let challenge = await groupLeader.post('/challenges', { | ||||
|     it('does not add challenge to creator\'s challenges', async () => { | ||||
|       await groupLeader.post('/challenges', { | ||||
|         group: group._id, | ||||
|         name: 'Test Challenge', | ||||
|         shortName: 'TC Label', | ||||
|       }); | ||||
|  | ||||
|       await expect(groupLeader.sync()).to.eventually.have.property('challenges').to.include(challenge._id); | ||||
|       await groupLeader.sync(); | ||||
|       expect(groupLeader.challenges.length).to.equal(0); | ||||
|     }); | ||||
|  | ||||
|     it('awards achievement if this is creator\'s first challenge', async () => { | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateChallenge, | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('POST /challenges/:challengeId/join', () => { | ||||
| @@ -43,6 +43,7 @@ describe('POST /challenges/:challengeId/join', () => { | ||||
|       authorizedUser = populatedGroup.members[0]; | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|     }); | ||||
|  | ||||
|     it('returns an error when user doesn\'t have permissions to access the challenge', async () => { | ||||
| @@ -91,6 +92,7 @@ describe('POST /challenges/:challengeId/join', () => { | ||||
|     }); | ||||
|  | ||||
|     it('increases memberCount of challenge', async () => { | ||||
|       await challenge.sync(); | ||||
|       let oldMemberCount = challenge.memberCount; | ||||
|  | ||||
|       await authorizedUser.post(`/challenges/${challenge._id}/join`); | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateChallenge, | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('POST /challenges/:challengeId/leave', () => { | ||||
| @@ -48,6 +48,7 @@ describe('POST /challenges/:challengeId/leave', () => { | ||||
|       notInGroupLeavingUser = populatedGroup.members[2]; | ||||
|  | ||||
|       challenge = await generateChallenge(groupLeader, group); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|       taskText = 'A challenge task text'; | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import { | ||||
|   sleep, | ||||
|   checkExistence, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('POST /challenges/:challengeId/winner/:winnerId', () => { | ||||
| @@ -58,6 +58,7 @@ describe('POST /challenges/:challengeId/winner/:winnerId', () => { | ||||
|       challenge = await generateChallenge(groupLeader, group, { | ||||
|         prize: 1, | ||||
|       }); | ||||
|       await groupLeader.post(`/challenges/${challenge._id}/join`); | ||||
|  | ||||
|       await groupLeader.post(`/tasks/challenge/${challenge._id}`, [ | ||||
|         {type: 'habit', text: taskText}, | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   generateUser, | ||||
|   generateGroup, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
|  | ||||
| describe('POST /challenges/:challengeId/clone', () => { | ||||
|   it('clones a challenge', async () => { | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { | ||||
|   generateChallenge, | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
|  | ||||
| describe('PUT /challenges/:challengeId', () => { | ||||
|   let privateGuild, user, nonMember, challenge, member; | ||||
| @@ -25,6 +25,7 @@ describe('PUT /challenges/:challengeId', () => { | ||||
|     member = members[0]; | ||||
|  | ||||
|     challenge = await generateChallenge(user, group); | ||||
|     await user.post(`/challenges/${challenge._id}/join`); | ||||
|     await member.post(`/challenges/${challenge._id}/join`); | ||||
|   }); | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   createAndPopulateGroup, | ||||
|   generateUser, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
| describe('DELETE /groups/:groupId/chat/:chatId', () => { | ||||
| @@ -53,16 +53,26 @@ describe('DELETE /groups/:groupId/chat/:chatId', () => { | ||||
|  | ||||
|     it('allows creator to delete a their message', async () => { | ||||
|       await user.del(`/groups/${groupWithChat._id}/chat/${nextMessage.id}`); | ||||
|       let messages = await user.get(`/groups/${groupWithChat._id}/chat/`); | ||||
|       expect(messages).is.an('array'); | ||||
|       expect(messages).to.not.include(nextMessage); | ||||
|  | ||||
|       const returnedMessages = await user.get(`/groups/${groupWithChat._id}/chat/`); | ||||
|       const messageFromUser = returnedMessages.find(returnedMessage => { | ||||
|         return returnedMessage.id === nextMessage.id; | ||||
|       }); | ||||
|  | ||||
|       expect(returnedMessages).is.an('array'); | ||||
|       expect(messageFromUser).to.not.exist; | ||||
|     }); | ||||
|  | ||||
|     it('allows admin to delete another user\'s message', async () => { | ||||
|       await admin.del(`/groups/${groupWithChat._id}/chat/${nextMessage.id}`); | ||||
|       let messages = await user.get(`/groups/${groupWithChat._id}/chat/`); | ||||
|       expect(messages).is.an('array'); | ||||
|       expect(messages).to.not.include(nextMessage); | ||||
|  | ||||
|       const returnedMessages = await user.get(`/groups/${groupWithChat._id}/chat/`); | ||||
|       const messageFromUser = returnedMessages.find(returnedMessage => { | ||||
|         return returnedMessage.id === nextMessage.id; | ||||
|       }); | ||||
|  | ||||
|       expect(returnedMessages).is.an('array'); | ||||
|       expect(messageFromUser).to.not.exist; | ||||
|     }); | ||||
|  | ||||
|     it('returns empty when previous message parameter is passed and the last message was deleted', async () => { | ||||
| @@ -71,9 +81,9 @@ describe('DELETE /groups/:groupId/chat/:chatId', () => { | ||||
|     }); | ||||
|  | ||||
|     it('returns the update chat when previous message parameter is passed and the chat is updated', async () => { | ||||
|       let deleteResult = await user.del(`/groups/${groupWithChat._id}/chat/${nextMessage.id}?previousMsg=${message.id}`); | ||||
|       const updatedChat = await user.del(`/groups/${groupWithChat._id}/chat/${nextMessage.id}?previousMsg=${message.id}`); | ||||
|  | ||||
|       expect(deleteResult[0].id).to.eql(message.id); | ||||
|       expect(updatedChat[0].id).to.eql(message.id); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   generateUser, | ||||
|   generateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
|  | ||||
| describe('GET /groups/:groupId/chat', () => { | ||||
|   let user; | ||||
| @@ -23,16 +23,17 @@ describe('GET /groups/:groupId/chat', () => { | ||||
|         privacy: 'public', | ||||
|       }, { | ||||
|         chat: [ | ||||
|           {text: 'Hello', flags: {}}, | ||||
|           {text: 'Welcome to the Guild', flags: {}}, | ||||
|           {text: 'Hello', flags: {}, id: 1}, | ||||
|           {text: 'Welcome to the Guild', flags: {}, id: 2}, | ||||
|         ], | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     it('returns Guild chat', async () => { | ||||
|       let chat = await user.get(`/groups/${group._id}/chat`); | ||||
|       const chat = await user.get(`/groups/${group._id}/chat`); | ||||
|  | ||||
|       expect(chat).to.eql(group.chat); | ||||
|       expect(chat[0].id).to.eql(group.chat[0].id); | ||||
|       expect(chat[1].id).to.eql(group.chat[1].id); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   generateUser, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { find } from 'lodash'; | ||||
|  | ||||
| describe('POST /chat/:chatId/flag', () => { | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   createAndPopulateGroup, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { find } from 'lodash'; | ||||
|  | ||||
| describe('POST /chat/:chatId/like', () => { | ||||
|   | ||||
| @@ -4,14 +4,14 @@ import { | ||||
|   translate as t, | ||||
|   sleep, | ||||
|   server, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import { | ||||
|   SPAM_MESSAGE_LIMIT, | ||||
|   SPAM_MIN_EXEMPT_CONTRIB_LEVEL, | ||||
|   TAVERN_ID, | ||||
| } from '../../../../../website/server/models/group'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
| import { getMatchesByWordArray, removePunctuationFromString } from '../../../../../website/server/libs/stringUtils'; | ||||
| import { getMatchesByWordArray } from '../../../../../website/server/libs/stringUtils'; | ||||
| import bannedWords from '../../../../../website/server/libs/bannedWords'; | ||||
| import guildsAllowingBannedWords from '../../../../../website/server/libs/guildsAllowingBannedWords'; | ||||
| import * as email from '../../../../../website/server/libs/email'; | ||||
| @@ -23,11 +23,11 @@ const BASE_URL = nconf.get('BASE_URL'); | ||||
| describe('POST /chat', () => { | ||||
|   let user, groupWithChat, member, additionalMember; | ||||
|   let testMessage = 'Test Message'; | ||||
|   let testBannedWordMessage = 'TEST_PLACEHOLDER_SWEAR_WORD_HERE'; | ||||
|   let testSlurMessage = 'message with TEST_PLACEHOLDER_SLUR_WORD_HERE'; | ||||
|   let bannedWordErrorMessage = t('bannedWordUsed').split('.'); | ||||
|   bannedWordErrorMessage[0] += ` (${removePunctuationFromString(testBannedWordMessage.toLowerCase())})`; | ||||
|   bannedWordErrorMessage = bannedWordErrorMessage.join('.'); | ||||
|   let testBannedWordMessage = 'TESTPLACEHOLDERSWEARWORDHERE'; | ||||
|   let testBannedWordMessage1 = 'TESTPLACEHOLDERSWEARWORDHERE1'; | ||||
|   let testSlurMessage = 'message with TESTPLACEHOLDERSLURWORDHERE'; | ||||
|   let testSlurMessage1 = 'TESTPLACEHOLDERSLURWORDHERE1'; | ||||
|   let bannedWordErrorMessage = t('bannedWordUsed', {swearWordsUsed: testBannedWordMessage}); | ||||
|  | ||||
|   before(async () => { | ||||
|     let { group, groupLeader, members } = await createAndPopulateGroup({ | ||||
| @@ -39,6 +39,7 @@ describe('POST /chat', () => { | ||||
|       members: 2, | ||||
|     }); | ||||
|     user = groupLeader; | ||||
|     await user.update({'contributor.level': SPAM_MIN_EXEMPT_CONTRIB_LEVEL}); // prevent tests accidentally throwing messageGroupChatSpam | ||||
|     groupWithChat = group; | ||||
|     member = members[0]; | ||||
|     additionalMember = members[1]; | ||||
| @@ -136,9 +137,19 @@ describe('POST /chat', () => { | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     it('checks error message has the banned words used', async () => { | ||||
|       let randIndex = Math.floor(Math.random() * (bannedWords.length + 1)); | ||||
|       let testBannedWords = bannedWords.slice(randIndex, randIndex + 2).map((w) => w.replace(/\\/g, '')); | ||||
|     it('errors when word is typed in mixed case', async () => { | ||||
|       let substrLength = Math.floor(testBannedWordMessage.length / 2); | ||||
|       let chatMessage = testBannedWordMessage.substring(0, substrLength).toLowerCase() + testBannedWordMessage.substring(substrLength).toUpperCase(); | ||||
|       await expect(user.post('/groups/habitrpg/chat', { message: chatMessage })) | ||||
|         .to.eventually.be.rejected.and.eql({ | ||||
|           code: 400, | ||||
|           error: 'BadRequest', | ||||
|           message: t('bannedWordUsed', {swearWordsUsed: chatMessage}), | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     it('checks error message has all the banned words used, regardless of case', async () => { | ||||
|       let testBannedWords = [testBannedWordMessage.toUpperCase(), testBannedWordMessage1.toLowerCase()]; | ||||
|       let chatMessage = `Mixing ${testBannedWords[0]} and ${testBannedWords[1]} is bad for you.`; | ||||
|       await expect(user.post('/groups/habitrpg/chat', { message: chatMessage})) | ||||
|         .to.eventually.be.rejected | ||||
| @@ -320,6 +331,17 @@ describe('POST /chat', () => { | ||||
|       members[0].flags.chatRevoked = false; | ||||
|       await members[0].update({'flags.chatRevoked': false}); | ||||
|     }); | ||||
|  | ||||
|     it('errors when slur is typed in mixed case', async () => { | ||||
|       let substrLength = Math.floor(testSlurMessage1.length / 2); | ||||
|       let chatMessage = testSlurMessage1.substring(0, substrLength).toLowerCase() + testSlurMessage1.substring(substrLength).toUpperCase(); | ||||
|       await expect(user.post('/groups/habitrpg/chat', { message: chatMessage })) | ||||
|         .to.eventually.be.rejected.and.eql({ | ||||
|           code: 400, | ||||
|           error: 'BadRequest', | ||||
|           message: t('bannedSlurUsed'), | ||||
|         }); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   it('does not error when sending a message to a private guild with a user with revoked chat', async () => { | ||||
| @@ -359,9 +381,28 @@ describe('POST /chat', () => { | ||||
|   }); | ||||
|  | ||||
|   it('creates a chat', async () => { | ||||
|     let message = await user.post(`/groups/${groupWithChat._id}/chat`, { message: testMessage}); | ||||
|     const newMessage = await user.post(`/groups/${groupWithChat._id}/chat`, { message: testMessage}); | ||||
|     const groupMessages = await user.get(`/groups/${groupWithChat._id}/chat`); | ||||
|  | ||||
|     expect(message.message.id).to.exist; | ||||
|     expect(newMessage.message.id).to.exist; | ||||
|     expect(groupMessages[0].id).to.exist; | ||||
|   }); | ||||
|  | ||||
|   it('creates a chat with a max length of 3000 chars', async () => { | ||||
|     const veryLongMessage = ` | ||||
|     123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789 123456789. | ||||
|     THIS PART WON'T BE IN THE MESSAGE (over 3000) | ||||
|     `; | ||||
|  | ||||
|     const newMessage = await user.post(`/groups/${groupWithChat._id}/chat`, { message: veryLongMessage}); | ||||
|     const groupMessages = await user.get(`/groups/${groupWithChat._id}/chat`); | ||||
|  | ||||
|     expect(newMessage.message.id).to.exist; | ||||
|     expect(groupMessages[0].id).to.exist; | ||||
|  | ||||
|     expect(newMessage.message.text.length).to.eql(3000); | ||||
|     expect(newMessage.message.text).to.not.contain('MESSAGE'); | ||||
|     expect(groupMessages[0].text.length).to.eql(3000); | ||||
|   }); | ||||
|  | ||||
|   it('creates a chat with user styles', async () => { | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   createAndPopulateGroup, | ||||
|   sleep, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
|  | ||||
| describe('POST /groups/:id/chat/seen', () => { | ||||
|   context('Guild', () => { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import { | ||||
|   createAndPopulateGroup, | ||||
|   generateUser, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import config from '../../../../../config.json'; | ||||
| import { v4 as generateUUID } from 'uuid'; | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { | ||||
|   requester, | ||||
|   translate as t, | ||||
| } from '../../../../helpers/api-v3-integration.helper'; | ||||
| } from '../../../../helpers/api-integration/v3'; | ||||
| import i18n from '../../../../../website/common/script/i18n'; | ||||
|  | ||||
| describe('GET /content', () => { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user