Compare commits
	
		
			494 Commits
		
	
	
		
			version/0.
			...
			version/0.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 019a0cb14d | |||
| 97290755e7 | |||
| 7f150c96b4 | |||
| 73558f30d1 | |||
| dfcfd87644 | |||
| 2c0f0a68a8 | |||
| 3d73aac3ab | |||
| e4fbcd3735 | |||
| 44c0eb37cf | |||
| adc3dcc2c4 | |||
| bac8227371 | |||
| 73d4d9dfe0 | |||
| afdac5f3f8 | |||
| dabce36667 | |||
| 3bd56ce522 | |||
| 540419d5c1 | |||
| ed1fcc3930 | |||
| c22ddc5394 | |||
| 0544864a3f | |||
| 0b9fc9e444 | |||
| e862b97005 | |||
| cffe09b02e | |||
| 846a86fb62 | |||
| 463c130351 | |||
| ffca957838 | |||
| 543e949a48 | |||
| feb80049aa | |||
| 5c59c8ccb6 | |||
| 1fadd82c65 | |||
| 7e7736126d | |||
| 5e0915afce | |||
| bf6c9e8c4a | |||
| 3353aa0298 | |||
| d4cb1a98c7 | |||
| 13f4ea0b8b | |||
| 261d57ad7b | |||
| 4086252979 | |||
| 8bdf12cff1 | |||
| 65a065c4ee | |||
| a691ee529c | |||
| f1c4a62612 | |||
| 358e39ced0 | |||
| 48c3f68cfc | |||
| 1849a7c383 | |||
| 82d14f37c3 | |||
| a0261eafa3 | |||
| 2a27325dfd | |||
| a6dee2e8ed | |||
| 2ff1635696 | |||
| 1cb6b5e984 | |||
| 1fe420fd80 | |||
| 50172e58d8 | |||
| d7483d129f | |||
| 34ed0b3594 | |||
| f008a3e20c | |||
| 9de950220f | |||
| 567c90b4c6 | |||
| ae19236366 | |||
| f9babe7089 | |||
| 78c74cd469 | |||
| 32abb27e61 | |||
| 8478b03892 | |||
| e972f2b289 | |||
| 22c4fb1414 | |||
| 0154def916 | |||
| fc69b6851d | |||
| 44a3c7fa5f | |||
| 37111fd07b | |||
| 4e6653e299 | |||
| 143a575369 | |||
| c782585287 | |||
| 7718b3b3b8 | |||
| 8ff9e72972 | |||
| ef6ef68a39 | |||
| 48a04744e0 | |||
| 6446ca8bb2 | |||
| b9991465ee | |||
| 3d8242be06 | |||
| 344a8817c3 | |||
| 3afb0d4f6d | |||
| c9714893bb | |||
| 3185a86b22 | |||
| a53f7a49ac | |||
| ca3bcc565d | |||
| 432176ea2f | |||
| c1dae0b599 | |||
| e70d3b6286 | |||
| 17e6bc921b | |||
| 46111e7cac | |||
| 3b7e47dbe2 | |||
| fff99f0e3d | |||
| 2e15b24f0a | |||
| 088b9592cd | |||
| b1e4e32b83 | |||
| d91a852eda | |||
| 171c5b9759 | |||
| 64290b2a37 | |||
| 72769b8a0a | |||
| 1018309413 | |||
| 6d0ecd228e | |||
| 40a651e66c | |||
| a390bb7b59 | |||
| 245ec65cbb | |||
| 17eea4a10c | |||
| 862fb0f5d2 | |||
| ec73b53340 | |||
| 9110f7fee3 | |||
| 54cc1fdeef | |||
| 8f42a7f0b4 | |||
| 2c221ea819 | |||
| 93e0441b58 | |||
| 7f1455cb12 | |||
| 59fc223a85 | |||
| 0a6f555c23 | |||
| 6a4233d6fd | |||
| 15fa7e9652 | |||
| f2acc154cd | |||
| d21ec6c9a5 | |||
| 43dd858cd5 | |||
| 34cbf5f702 | |||
| 3c6e94b6a8 | |||
| 1cd149c815 | |||
| 4c6f562805 | |||
| e59c4ec1c7 | |||
| 1169db7530 | |||
| 1453008796 | |||
| 2209b6d603 | |||
| ccbc0384f9 | |||
| a48924c896 | |||
| dc8d8dd2b6 | |||
| afca94ceb8 | |||
| 0b86231a36 | |||
| c0df1f38b8 | |||
| 2b8fed8f4e | |||
| c7322a32a0 | |||
| 64b75cab84 | |||
| f58bc61999 | |||
| fb8ccc0283 | |||
| c38012f147 | |||
| 3676ff21c2 | |||
| 920e705d75 | |||
| de0b137b1e | |||
| d44ac6e2a3 | |||
| 71039a4012 | |||
| 8745ac7932 | |||
| 7f70048423 | |||
| 97dbfc8885 | |||
| 149ea22a93 | |||
| 404ed5406d | |||
| b8656858ec | |||
| 6b0f0e8993 | |||
| aec1ccd88d | |||
| bee5c200b6 | |||
| 9d640efc88 | |||
| f0907841dd | |||
| 2bffc12ef9 | |||
| 2ff9ec6522 | |||
| 43a54f5c54 | |||
| 7bff2734aa | |||
| 84768c0ec6 | |||
| f4499a5459 | |||
| b3aede5bba | |||
| 531ea1c039 | |||
| c2c5ff6912 | |||
| 9cddab8fd5 | |||
| 06d15d8a27 | |||
| b5c711854b | |||
| 4cf6c36f34 | |||
| 75a6f6c875 | |||
| 62abe3f256 | |||
| 9296c41650 | |||
| 7fb48fde6d | |||
| 174472bb45 | |||
| 17575ed921 | |||
| b1b1a27444 | |||
| f97a5eeefb | |||
| 10fd96981e | |||
| 67e3eb549c | |||
| 30a6d1f0b1 | |||
| 3d1fa9f048 | |||
| 1d2be6e68b | |||
| c21e343986 | |||
| ff37ed095c | |||
| 8623a2c3fc | |||
| 23d277eaf1 | |||
| 75ced59451 | |||
| bccf424c5e | |||
| 2f9ae40d20 | |||
| 11e1eec3fb | |||
| 765c5633df | |||
| 6344b1aafb | |||
| ed25801e6e | |||
| 4d0148193f | |||
| 804ae15c2e | |||
| b35a9fad86 | |||
| a4f83bd28a | |||
| 796f83c3d0 | |||
| 2099bbb713 | |||
| 67beba8f78 | |||
| a798412e17 | |||
| 3b2c2d781f | |||
| 98c844f3d6 | |||
| 2645bd0132 | |||
| 2c4fc56b49 | |||
| 0ec1468058 | |||
| 5d1a3043b2 | |||
| b46958d1f9 | |||
| 5daa8d5fe3 | |||
| 31846f1d05 | |||
| 1fac964b8b | |||
| dfa6ed8ac2 | |||
| 66fe10299e | |||
| e0a3ec033f | |||
| 7033ec0ab9 | |||
| 4004579905 | |||
| 9fe9e48a5c | |||
| 595a6c7fe6 | |||
| 11b5860d4a | |||
| 9bdbff4cda | |||
| e0d597eeac | |||
| f576985cc9 | |||
| 22a6aef60b | |||
| ec0a6e7854 | |||
| 6904608e6f | |||
| cb3732cb2b | |||
| 57de6cbafc | |||
| b1dda764a9 | |||
| 5ec2102487 | |||
| 9f8fb7378a | |||
| 98cd646044 | |||
| 0cba1b4c45 | |||
| 53918462b6 | |||
| 8a7e74b523 | |||
| 4dc7065e97 | |||
| 3c93bb9f9f | |||
| 8143fae2d6 | |||
| 3cfe45d3cb | |||
| 8e5c3f2f31 | |||
| 5a3b2fdd49 | |||
| e47b9f0d57 | |||
| 146dd747f1 | |||
| f2ce56063b | |||
| b26f378e4c | |||
| 9072b836c6 | |||
| 2fa57d064e | |||
| 146705c60a | |||
| 5029a99df6 | |||
| e7129d18f6 | |||
| d2bf9f81d6 | |||
| 30acf0660b | |||
| dda41af5c8 | |||
| 9b5b03647b | |||
| 940b3eb943 | |||
| 16eb629b71 | |||
| 755045b226 | |||
| 61478db94e | |||
| f69f959bdb | |||
| 146edb45d4 | |||
| 045a802365 | |||
| c90d8ddcff | |||
| 3ff2ec929f | |||
| a3ef26b7ad | |||
| 19cd1624c1 | |||
| 366ef352c6 | |||
| a9031a6abc | |||
| a1a5223b58 | |||
| c723b0233f | |||
| b369eb28f1 | |||
| 9b8f390e31 | |||
| 11630c9a74 | |||
| c9ac10f6f6 | |||
| 04d613cb28 | |||
| 40866f9ecd | |||
| d8585eb872 | |||
| 15aaeda475 | |||
| 8536ef9e23 | |||
| 35b6bb6b3f | |||
| eaa573c715 | |||
| 660972e303 | |||
| a21012bf0c | |||
| 8dbafa4bda | |||
| 80049413f0 | |||
| 2739442d4a | |||
| c679f0a67c | |||
| d9a952dd03 | |||
| 9a1a0f0aa8 | |||
| 4d6bb60134 | |||
| 80e6d59382 | |||
| 81ac951872 | |||
| f33e553cfd | |||
| 9b0240dc26 | |||
| c327310392 | |||
| 457375287c | |||
| 7e87bfef5b | |||
| a7af5268de | |||
| 6d916029bb | |||
| 81fdcbadad | |||
| ec1e25fe71 | |||
| b5306e4a94 | |||
| 801b8a1e59 | |||
| 3a52059793 | |||
| 10b7d99b37 | |||
| 6be8d0cbb2 | |||
| 5b8e3689ec | |||
| 25a5d8f5da | |||
| 883d439544 | |||
| 1c3b5889e5 | |||
| 87012b65e1 | |||
| 29913773a7 | |||
| 0bc6a4fed4 | |||
| 4645d8353f | |||
| 260c5555fa | |||
| 6f7b917c38 | |||
| 1456ee6d3e | |||
| ae3d3d0295 | |||
| c23ceacd0b | |||
| 5155204283 | |||
| 5509ec9b0f | |||
| d6f9b2e47d | |||
| 67aa4aef11 | |||
| 9e46c8bfec | |||
| 1eaa9b9733 | |||
| ee05834b69 | |||
| fccc8f4959 | |||
| c721620f96 | |||
| c9f73d718e | |||
| bfa58be721 | |||
| 4bb602149e | |||
| 81ab9092fc | |||
| 29d5962c4c | |||
| 5c75339946 | |||
| 4774d9a46c | |||
| dbe16ba4fd | |||
| 6972cf00a0 | |||
| 0445be9712 | |||
| 89dbdd9585 | |||
| da88ce7150 | |||
| 5f50fcfcf5 | |||
| 96be087221 | |||
| a53a269a8c | |||
| 59565a5286 | |||
| ae3c092238 | |||
| e98e5e4e3e | |||
| d50c7ec8d4 | |||
| c0fdf377d1 | |||
| 70c11c8988 | |||
| 67b19becc1 | |||
| ae64024ef4 | |||
| e6571826cb | |||
| c621e61978 | |||
| 3626fa4b98 | |||
| 01b0eb159a | |||
| 63aa48d981 | |||
| 2e0ba05d55 | |||
| b2ac57bb67 | |||
| 4c22e5c2c8 | |||
| 4a7b0ec8a9 | |||
| 330118249e | |||
| 8d4dabde02 | |||
| cf7323c41b | |||
| edd856df7d | |||
| 5e35859db6 | |||
| acabb2df54 | |||
| e6376a05f7 | |||
| 1f45aff7ad | |||
| e1f1f617b6 | |||
| 2690675dca | |||
| 7529b51358 | |||
| c394066d99 | |||
| 9c585032ef | |||
| d408031304 | |||
| c47bc11ec0 | |||
| 1deb094afe | |||
| 501fed1922 | |||
| ad8125ac1c | |||
| b42a551fb2 | |||
| 3256be23df | |||
| f7c0c0146a | |||
| e4baf8c21e | |||
| 364f040b36 | |||
| 2b8c2b2346 | |||
| 5f861189e4 | |||
| 5e11b6687e | |||
| c4b429825d | |||
| eebbae0677 | |||
| 42b30f4507 | |||
| 0e425418df | |||
| 7fe0300b86 | |||
| c012c6be5c | |||
| a5dc193cfd | |||
| 7507ad2620 | |||
| f1291fec8d | |||
| 37aeeea239 | |||
| 0fa1fc86da | |||
| c3034ab9ac | |||
| 76694e037a | |||
| 787db41cc3 | |||
| 74da3df7cd | |||
| a6e435bd70 | |||
| c313b496aa | |||
| a7eaa74191 | |||
| 11ecdc4fcf | |||
| 2f7781b67a | |||
| 296d4f691a | |||
| 64033031b1 | |||
| 9daff7608d | |||
| 0a4af80b9b | |||
| a54adb05c4 | |||
| 43a389e596 | |||
| 2d7e8f1b50 | |||
| cf11f6b121 | |||
| 6dcdf7bcce | |||
| 56d872af15 | |||
| ca663d16fc | |||
| e05c18b19b | |||
| a7b86e46bc | |||
| 84f56674c2 | |||
| 02ab177c6d | |||
| 1232c487e9 | |||
| ef0a2bfbe8 | |||
| 05242a11ad | |||
| 4593ad7bcc | |||
| d7fd5a7fa6 | |||
| 4439378fd4 | |||
| acf65eafdd | |||
| c2ebff55ef | |||
| 99c82676b6 | |||
| 4991e9b825 | |||
| 612f95c3ba | |||
| cd91d5ca15 | |||
| cbbbb5dc08 | |||
| c1640b9411 | |||
| a4842c1f95 | |||
| a4707ddc54 | |||
| fb82d56307 | |||
| 1a1005f80d | |||
| e86cae6cac | |||
| 0b282f45e0 | |||
| 791e88ffc1 | |||
| 7bd3c4bccf | |||
| 722e2e4050 | |||
| c7fc444c95 | |||
| 20ad062814 | |||
| fcb5d36e07 | |||
| 9b131b619f | |||
| 54427f7c68 | |||
| 35eef9c28d | |||
| e88a82553d | |||
| 01a9520140 | |||
| 46667615c3 | |||
| c6721a83a4 | |||
| 46866e8ef0 | |||
| 4a49681127 | |||
| 4c3fced4e9 | |||
| 172347d90f | |||
| f54520b5cf | |||
| d7c4697625 | |||
| 5584f5bda8 | |||
| 2ce6f5a714 | |||
| c66945623a | |||
| cbae05c74c | |||
| 5b771da972 | |||
| 2db1738e4a | |||
| 95de6a14fd | |||
| 17132ebc19 | |||
| 289be46388 | |||
| 6c300b7b31 | |||
| b726583084 | |||
| 48055d1cfd | |||
| 436070f5bd | |||
| 3ee79818db | |||
| e7a02104db | |||
| 556740d7bc | |||
| 421f51770c | |||
| 96f7e70f9e | |||
| ad96f7dbb8 | |||
| e7fb48eba2 | |||
| b19b5b644d | |||
| 250b6691d4 | |||
| e3b02a6e78 | |||
| e94ef34d8f | |||
| 49e945307a | |||
| edfe0e5450 | |||
| 06b65a7882 | |||
| ff9bc8aa70 | |||
| 28da67abe6 | |||
| 39d9fe9bf0 | |||
| 750117b0fd | |||
| 983462f80d | |||
| 4ae31d409b | |||
| 98b414f3e2 | |||
| a0d42092e3 | |||
| f2569b6424 | |||
| 9d344d887c | 
| @ -1,44 +1,27 @@ | |||||||
| [bumpversion] | [bumpversion] | ||||||
| current_version = 0.0.8-alpha | current_version = 0.7.2-beta | ||||||
| tag = True | tag = True | ||||||
| commit = True | commit = True | ||||||
| parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)\-(?P<release>.*) | parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)\-(?P<release>.*) | ||||||
| serialize = {major}.{minor}.{patch}-{release} | serialize = {major}.{minor}.{patch}-{release} | ||||||
| message = bump version: {current_version} -> {new_version} | message = new release: {new_version} | ||||||
| tag_name = version/{new_version} | tag_name = version/{new_version} | ||||||
|  |  | ||||||
| [bumpversion:part:release] | [bumpversion:part:release] | ||||||
| optional_value = stable | optional_value = stable | ||||||
|  | first_value = beta | ||||||
| values =  | values =  | ||||||
| 	alpha | 	alpha | ||||||
| 	beta | 	beta | ||||||
| 	stable | 	stable | ||||||
|  |  | ||||||
|  | [bumpversion:file:helm/passbook/values.yaml] | ||||||
|  |  | ||||||
| [bumpversion:file:helm/passbook/Chart.yaml] | [bumpversion:file:helm/passbook/Chart.yaml] | ||||||
|  |  | ||||||
| [bumpversion:file:.gitlab-ci.yml] | [bumpversion:file:.gitlab-ci.yml] | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/__init__.py] | [bumpversion:file:passbook/__init__.py] | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/api/__init__.py] | [bumpversion:file:docker/nginx.conf] | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/core/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/admin/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/captcha_factor/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/oauth_client/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/ldap/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/lib/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/saml_idp/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/audit/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/oauth_provider/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/otp/__init__.py] |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,12 +1,10 @@ | |||||||
| [run] | [run] | ||||||
| source = passbook | source = passbook | ||||||
| omit = | omit = | ||||||
|     env/ |  | ||||||
|     */wsgi.py |     */wsgi.py | ||||||
|     manage.py |     manage.py | ||||||
|     */migrations/* |     */migrations/* | ||||||
|     */apps.py |     */apps.py | ||||||
|     passbook/management/commands/nexus_upload.py |  | ||||||
|     passbook/management/commands/web.py |     passbook/management/commands/web.py | ||||||
|     passbook/management/commands/worker.py |     passbook/management/commands/worker.py | ||||||
|     docs/ |     docs/ | ||||||
| @ -23,6 +21,7 @@ exclude_lines = | |||||||
|     def __str__ |     def __str__ | ||||||
|     def __repr__ |     def __repr__ | ||||||
|     if self\.debug |     if self\.debug | ||||||
|  |     if TYPE_CHECKING | ||||||
|  |  | ||||||
|     # Don't complain if tests don't hit defensive assertion code: |     # Don't complain if tests don't hit defensive assertion code: | ||||||
|     raise AssertionError |     raise AssertionError | ||||||
|  | |||||||
| @ -2,3 +2,4 @@ env | |||||||
| helm | helm | ||||||
| passbook-ui | passbook-ui | ||||||
| static | static | ||||||
|  | *.env.yml | ||||||
|  | |||||||
| @ -9,3 +9,6 @@ insert_final_newline = true | |||||||
|  |  | ||||||
| [html] | [html] | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  |  | ||||||
|  | [yaml] | ||||||
|  | indent_size = 2 | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -191,3 +191,8 @@ pip-selfcheck.json | |||||||
| # End of https://www.gitignore.io/api/python,django | # End of https://www.gitignore.io/api/python,django | ||||||
| /static/ | /static/ | ||||||
| local.env.yml | local.env.yml | ||||||
|  | .vscode/ | ||||||
|  |  | ||||||
|  | ### Helm ### | ||||||
|  | # Chart dependencies | ||||||
|  | **/charts/*.tgz | ||||||
|  | |||||||
							
								
								
									
										210
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							
							
						
						
									
										210
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							| @ -1,140 +1,152 @@ | |||||||
| # Global Variables | # Global Variables | ||||||
| before_script: |  | ||||||
|   - "python3 -m pip install -U virtualenv" |  | ||||||
|   - "virtualenv env" |  | ||||||
|   - "source env/bin/activate" |  | ||||||
|   - "pip3 install -U -r requirements-dev.txt" |  | ||||||
| stages: | stages: | ||||||
|  |   - build-base-image | ||||||
|  |   - build-dev-image | ||||||
|   - test |   - test | ||||||
|   - build |   - build | ||||||
|   - docs |   - package | ||||||
| image: python:3.6 |   - post-release | ||||||
| services: | image: docker.beryju.org/passbook/dev:latest | ||||||
|   - postgres:latest |  | ||||||
|  |  | ||||||
| variables: | variables: | ||||||
|   POSTGRES_DB: passbook |   POSTGRES_DB: passbook | ||||||
|   POSTGRES_USER: passbook |   POSTGRES_USER: passbook | ||||||
|   POSTGRES_PASSWORD: 'EK-5jnKfjrGRm<77' |   POSTGRES_PASSWORD: "EK-5jnKfjrGRm<77" | ||||||
|   SUPERVISR_ENV: ci |  | ||||||
|  | before_script: | ||||||
|  |   - pip install pipenv | ||||||
|  |   # Ensure all dependencies are installed, even those not included in passbook/dev | ||||||
|  |   # According to pipenv docs, -d outputs all packages, however it actually does not | ||||||
|  |   - pipenv lock -r > requirements-all.txt | ||||||
|  |   - pipenv lock -rd >> requirements-all.txt | ||||||
|  |   - pip install -r requirements-all.txt | ||||||
|  |  | ||||||
|  | create-base-image: | ||||||
|  |   image: | ||||||
|  |     name: gcr.io/kaniko-project/executor:debug | ||||||
|  |     entrypoint: [""] | ||||||
|  |   before_script: | ||||||
|  |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|  |   script: | ||||||
|  |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/base.Dockerfile --destination docker.beryju.org/passbook/base:latest | ||||||
|  |   stage: build-base-image | ||||||
|  |   only: | ||||||
|  |     refs: | ||||||
|  |       - tags | ||||||
|  |       - /^version/.*$/ | ||||||
|  |  | ||||||
|  | build-dev-image: | ||||||
|  |   image: | ||||||
|  |     name: gcr.io/kaniko-project/executor:debug | ||||||
|  |     entrypoint: [""] | ||||||
|  |   before_script: | ||||||
|  |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|  |   script: | ||||||
|  |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/dev.Dockerfile --destination docker.beryju.org/passbook/dev:latest | ||||||
|  |   stage: build-dev-image | ||||||
|  |   only: | ||||||
|  |     refs: | ||||||
|  |       - tags | ||||||
|  |       - /^version/.*$/ | ||||||
|  |  | ||||||
| include: |  | ||||||
|   - /allauth/.gitlab-ci.yml |  | ||||||
|  |  | ||||||
| isort: | isort: | ||||||
|   script: |   script: | ||||||
|     - isort -c -sg env |     - isort -c -sg env | ||||||
|   stage: test |   stage: test | ||||||
|  |   services: | ||||||
|  |   - postgres:latest | ||||||
|  |   - redis:latest | ||||||
| migrations: | migrations: | ||||||
|   script: |   script: | ||||||
|     - python manage.py migrate |     - python manage.py migrate | ||||||
|   stage: test |   stage: test | ||||||
| prospector: |   services: | ||||||
|   script: |   - postgres:latest | ||||||
|     - prospector |   - redis:latest | ||||||
|   stage: test | # prospector: | ||||||
|  | #   script: | ||||||
|  | #     - prospector | ||||||
|  | #   stage: test | ||||||
|  | #   services: | ||||||
|  | #   - postgres:latest | ||||||
|  | #   - redis:latest | ||||||
| pylint: | pylint: | ||||||
|   script: |   script: | ||||||
|     - pylint passbook |     - pylint passbook | ||||||
|   stage: test |   stage: test | ||||||
|  |   services: | ||||||
|  |   - postgres:latest | ||||||
|  |   - redis:latest | ||||||
| coverage: | coverage: | ||||||
|   script: |   script: | ||||||
|     - coverage run manage.py test |     - coverage run --concurrency=multiprocessing manage.py test | ||||||
|  |     - coverage combine | ||||||
|     - coverage report |     - coverage report | ||||||
|   stage: test |   stage: test | ||||||
| bandit: |   services: | ||||||
|   script: |   - postgres:latest | ||||||
|     - bandit -r passbook |   - redis:latest | ||||||
|   stage: test |  | ||||||
|  |  | ||||||
| package-docker: | build-passbook-server: | ||||||
|  |   stage: build | ||||||
|   image: |   image: | ||||||
|     name: gcr.io/kaniko-project/executor:debug |     name: gcr.io/kaniko-project/executor:debug | ||||||
|     entrypoint: [""] |     entrypoint: [""] | ||||||
|   before_script: |   before_script: | ||||||
|     - echo "{\"auths\":{\"https://docker.$NEXUS_URL/\":{\"username\":\"$NEXUS_USER\",\"password\":\"$NEXUS_PASS\"}}}" > /kaniko/.docker/config.json |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|   script: |   script: | ||||||
|     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/Dockerfile --destination docker.pkg.beryju.org/passbook:latest --destination docker.pkg.beryju.org/passbook:0.0.8-alpha |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/Dockerfile --destination docker.beryju.org/passbook/server:latest --destination docker.beryju.org/passbook/server:0.7.2-beta | ||||||
|   stage: build |  | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
|     - /^version/.*$/ |     - /^version/.*$/ | ||||||
| package-helm: | build-passbook-static: | ||||||
|   stage: build |   stage: build | ||||||
|  |   image: | ||||||
|  |     name: gcr.io/kaniko-project/executor:debug | ||||||
|  |     entrypoint: [""] | ||||||
|  |   before_script: | ||||||
|  |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|   script: |   script: | ||||||
|     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/static.Dockerfile --destination docker.beryju.org/passbook/static:latest --destination docker.beryju.org/passbook/static:0.7.2-beta | ||||||
|     - helm init --client-only |  | ||||||
|     - helm package helm/passbook |  | ||||||
|     - ./manage.py nexus_upload --method put --url $NEXUS_URL --user $NEXUS_USER --password $NEXUS_PASS --repo helm *.tgz |  | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
|     - /^version/.*$/ |     - /^version/.*$/ | ||||||
| # package-3.5: |   # running collectstatic fully initialises django, hence we need that databases | ||||||
| #   before_script: |   services: | ||||||
| #     - apt update |     - postgres:latest | ||||||
| #     - apt install -y build-essential debhelper devscripts equivs python3 python3-pip |     - redis:latest | ||||||
| #     - cp debian/control-3.5 debian/control |  | ||||||
| #     - mk-build-deps debian/control |  | ||||||
| #     - apt install ./*build-deps*deb -f -y |  | ||||||
| #     - "python3 -m pip install -U virtualenv" |  | ||||||
| #     - "virtualenv env" |  | ||||||
| #     - "source env/bin/activate" |  | ||||||
| #     - "pip3 install -U -r requirements.txt -r requirements-dev.txt" |  | ||||||
| #   image: debian |  | ||||||
| #   script: |  | ||||||
| #     - debuild -us -uc |  | ||||||
| #     - cp ../passbook*.deb . |  | ||||||
| #     - python manage.py nexus_upload |  | ||||||
| #   artifacts: |  | ||||||
| #     paths: |  | ||||||
| #     - passbook-python3.5*deb |  | ||||||
| #     expire_in: 2 days |  | ||||||
| #   stage: build |  | ||||||
| #   only: |  | ||||||
| #   - tags |  | ||||||
| #   - /^debian/.*$/ |  | ||||||
| # package-3.6: |  | ||||||
| #   before_script: |  | ||||||
| #     - apt update |  | ||||||
| #     - apt install -y build-essential debhelper devscripts equivs python3 python3-pip |  | ||||||
| #     - cp debian/control-3.6 debian/control |  | ||||||
| #     - mk-build-deps debian/control |  | ||||||
| #     - apt install ./*build-deps*deb -f -y |  | ||||||
| #     - "python3 -m pip install -U virtualenv" |  | ||||||
| #     - "virtualenv env" |  | ||||||
| #     - "source env/bin/activate" |  | ||||||
| #     - "pip3 install -U -r requirements.txt -r requirements-dev.txt" |  | ||||||
| #   image: debian:buster |  | ||||||
| #   script: |  | ||||||
| #     - debuild -us -uc |  | ||||||
| #     - cp ../passbook*.deb . |  | ||||||
| #     - python manage.py nexus_upload |  | ||||||
| #   artifacts: |  | ||||||
| #     paths: |  | ||||||
| #     - passbook-python3.6*deb |  | ||||||
| #     expire_in: 2 days |  | ||||||
| #   stage: build |  | ||||||
| #   only: |  | ||||||
| #     - tags |  | ||||||
| #     - /^debian/.*$r |  | ||||||
|  |  | ||||||
| # docs: | package-helm: | ||||||
| #   stage: docs |   image: debian:stretch-slim | ||||||
| #   only: |   stage: package | ||||||
| #   - master |   before_script: | ||||||
| #   - tags |     - apt update && apt install -y curl | ||||||
| #   - /^debian/.*$/ |     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||||
| #   environment: |   script: | ||||||
| #     name: docs |     - helm init --client-only | ||||||
| #     url: "https://passbook.beryju.org/docs/" |     - helm dependency update helm/passbook | ||||||
| #   script: |     - helm package helm/passbook | ||||||
| #     - apt update |   artifacts: | ||||||
| #     - apt install -y rsync |     paths: | ||||||
| #     - "mkdir ~/.ssh" |       - passbook-*.tgz | ||||||
| #     - "cp .gitlab/known_hosts ~/.ssh/" |     expire_in: 1 week | ||||||
| #     - "pip3 install -U -r requirements-docs.txt" |   only: | ||||||
| #     - "eval $(ssh-agent -s)" |     - tags | ||||||
| #     - "echo \"${CI_SSH_PRIVATE}\" | ssh-add -" |     - /^version/.*$/ | ||||||
| #     - mkdocs build |  | ||||||
| #     - 'rsync -avh --delete web/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/"' | notify-sentry: | ||||||
| #     - 'rsync -avh --delete site/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/docs/"' |   image: getsentry/sentry-cli | ||||||
|  |   stage: post-release | ||||||
|  |   variables: | ||||||
|  |     SENTRY_URL: https://sentry.beryju.org | ||||||
|  |     SENTRY_ORG: beryjuorg | ||||||
|  |     SENTRY_PROJECT: passbook | ||||||
|  |   before_script: | ||||||
|  |     - apk add curl | ||||||
|  |   script: | ||||||
|  |     - sentry-cli releases new passbook@0.7.2-beta | ||||||
|  |     - sentry-cli releases set-commits --auto passbook@0.7.2-beta | ||||||
|  |   only: | ||||||
|  |     - tags | ||||||
|  |     - /^version/.*$/ | ||||||
|  | |||||||
| @ -3,7 +3,6 @@ test-warnings: true | |||||||
| doc-warnings: false | doc-warnings: false | ||||||
|  |  | ||||||
| ignore-paths: | ignore-paths: | ||||||
|   - env |  | ||||||
|   - migrations |   - migrations | ||||||
|   - docs |   - docs | ||||||
|   - node_modules |   - node_modules | ||||||
|  | |||||||
| @ -2,9 +2,10 @@ | |||||||
|  |  | ||||||
| disable=redefined-outer-name,arguments-differ,no-self-use,cyclic-import,fixme,locally-disabled,unpacking-non-sequence,too-many-ancestors,too-many-branches,too-few-public-methods | disable=redefined-outer-name,arguments-differ,no-self-use,cyclic-import,fixme,locally-disabled,unpacking-non-sequence,too-many-ancestors,too-many-branches,too-few-public-methods | ||||||
| load-plugins=pylint_django,pylint.extensions.bad_builtin | load-plugins=pylint_django,pylint.extensions.bad_builtin | ||||||
| #,pylint.extensions.docparams |  | ||||||
| extension-pkg-whitelist=lxml | extension-pkg-whitelist=lxml | ||||||
| const-rgx=[a-zA-Z0-9_]{1,40}$ | const-rgx=[a-zA-Z0-9_]{1,40}$ | ||||||
|  | ignored-modules=django-otp | ||||||
|  | jobs=4 | ||||||
|  |  | ||||||
| [SIMILARITIES] | [SIMILARITIES] | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										114
									
								
								.vscode/.ropeproject/config.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										114
									
								
								.vscode/.ropeproject/config.py
									
									
									
									
										vendored
									
									
								
							| @ -1,114 +0,0 @@ | |||||||
| # The default ``config.py`` |  | ||||||
| # flake8: noqa |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def set_prefs(prefs): |  | ||||||
|     """This function is called before opening the project""" |  | ||||||
|  |  | ||||||
|     # Specify which files and folders to ignore in the project. |  | ||||||
|     # Changes to ignored resources are not added to the history and |  | ||||||
|     # VCSs.  Also they are not returned in `Project.get_files()`. |  | ||||||
|     # Note that ``?`` and ``*`` match all characters but slashes. |  | ||||||
|     # '*.pyc': matches 'test.pyc' and 'pkg/test.pyc' |  | ||||||
|     # 'mod*.pyc': matches 'test/mod1.pyc' but not 'mod/1.pyc' |  | ||||||
|     # '.svn': matches 'pkg/.svn' and all of its children |  | ||||||
|     # 'build/*.o': matches 'build/lib.o' but not 'build/sub/lib.o' |  | ||||||
|     # 'build//*.o': matches 'build/lib.o' and 'build/sub/lib.o' |  | ||||||
|     prefs['ignored_resources'] = ['*.pyc', '*~', '.ropeproject', |  | ||||||
|                                   '.hg', '.svn', '_svn', '.git', '.tox'] |  | ||||||
|  |  | ||||||
|     # Specifies which files should be considered python files.  It is |  | ||||||
|     # useful when you have scripts inside your project.  Only files |  | ||||||
|     # ending with ``.py`` are considered to be python files by |  | ||||||
|     # default. |  | ||||||
|     # prefs['python_files'] = ['*.py'] |  | ||||||
|  |  | ||||||
|     # Custom source folders:  By default rope searches the project |  | ||||||
|     # for finding source folders (folders that should be searched |  | ||||||
|     # for finding modules).  You can add paths to that list.  Note |  | ||||||
|     # that rope guesses project source folders correctly most of the |  | ||||||
|     # time; use this if you have any problems. |  | ||||||
|     # The folders should be relative to project root and use '/' for |  | ||||||
|     # separating folders regardless of the platform rope is running on. |  | ||||||
|     # 'src/my_source_folder' for instance. |  | ||||||
|     # prefs.add('source_folders', 'src') |  | ||||||
|  |  | ||||||
|     # You can extend python path for looking up modules |  | ||||||
|     # prefs.add('python_path', '~/python/') |  | ||||||
|  |  | ||||||
|     # Should rope save object information or not. |  | ||||||
|     prefs['save_objectdb'] = True |  | ||||||
|     prefs['compress_objectdb'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope analyzes each module when it is being saved. |  | ||||||
|     prefs['automatic_soa'] = True |  | ||||||
|     # The depth of calls to follow in static object analysis |  | ||||||
|     prefs['soa_followed_calls'] = 0 |  | ||||||
|  |  | ||||||
|     # If `False` when running modules or unit tests "dynamic object |  | ||||||
|     # analysis" is turned off.  This makes them much faster. |  | ||||||
|     prefs['perform_doa'] = True |  | ||||||
|  |  | ||||||
|     # Rope can check the validity of its object DB when running. |  | ||||||
|     prefs['validate_objectdb'] = True |  | ||||||
|  |  | ||||||
|     # How many undos to hold? |  | ||||||
|     prefs['max_history_items'] = 32 |  | ||||||
|  |  | ||||||
|     # Shows whether to save history across sessions. |  | ||||||
|     prefs['save_history'] = True |  | ||||||
|     prefs['compress_history'] = False |  | ||||||
|  |  | ||||||
|     # Set the number spaces used for indenting.  According to |  | ||||||
|     # :PEP:`8`, it is best to use 4 spaces.  Since most of rope's |  | ||||||
|     # unit-tests use 4 spaces it is more reliable, too. |  | ||||||
|     prefs['indent_size'] = 4 |  | ||||||
|  |  | ||||||
|     # Builtin and c-extension modules that are allowed to be imported |  | ||||||
|     # and inspected by rope. |  | ||||||
|     prefs['extension_modules'] = [] |  | ||||||
|  |  | ||||||
|     # Add all standard c-extensions to extension_modules list. |  | ||||||
|     prefs['import_dynload_stdmods'] = True |  | ||||||
|  |  | ||||||
|     # If `True` modules with syntax errors are considered to be empty. |  | ||||||
|     # The default value is `False`; When `False` syntax errors raise |  | ||||||
|     # `rope.base.exceptions.ModuleSyntaxError` exception. |  | ||||||
|     prefs['ignore_syntax_errors'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope ignores unresolvable imports.  Otherwise, they |  | ||||||
|     # appear in the importing namespace. |  | ||||||
|     prefs['ignore_bad_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will insert new module imports as |  | ||||||
|     # `from <package> import <module>` by default. |  | ||||||
|     prefs['prefer_module_from_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will transform a comma list of imports into |  | ||||||
|     # multiple separate import statements when organizing |  | ||||||
|     # imports. |  | ||||||
|     prefs['split_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will remove all top-level import statements and |  | ||||||
|     # reinsert them at the top of the module when making changes. |  | ||||||
|     prefs['pull_imports_to_top'] = True |  | ||||||
|  |  | ||||||
|     # If `True`, rope will sort imports alphabetically by module name instead |  | ||||||
|     # of alphabetically by import statement, with from imports after normal |  | ||||||
|     # imports. |  | ||||||
|     prefs['sort_imports_alphabetically'] = False |  | ||||||
|  |  | ||||||
|     # Location of implementation of |  | ||||||
|     # rope.base.oi.type_hinting.interfaces.ITypeHintingFactory In general |  | ||||||
|     # case, you don't have to change this value, unless you're an rope expert. |  | ||||||
|     # Change this value to inject you own implementations of interfaces |  | ||||||
|     # listed in module rope.base.oi.type_hinting.providers.interfaces |  | ||||||
|     # For example, you can add you own providers for Django Models, or disable |  | ||||||
|     # the search type-hinting in a class hierarchy, etc. |  | ||||||
|     prefs['type_hinting_factory'] = ( |  | ||||||
|         'rope.base.oi.type_hinting.factory.default_type_hinting_factory') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def project_opened(project): |  | ||||||
|     """This function is called after opening the project""" |  | ||||||
|     # Do whatever you like here! |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								.vscode/.ropeproject/objectdb
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								.vscode/.ropeproject/objectdb
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										11
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @ -1,11 +0,0 @@ | |||||||
| { |  | ||||||
|   "python.pythonPath": "env/bin/python", |  | ||||||
|   "editor.tabSize": 4, |  | ||||||
|   "[html]": { |  | ||||||
|     "editor.tabSize": 2 |  | ||||||
|   }, |  | ||||||
|   "cSpell.words": [ |  | ||||||
|     "SAML", |  | ||||||
|     "passbook" |  | ||||||
|   ] |  | ||||||
| } |  | ||||||
							
								
								
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -1,28 +1,9 @@ | |||||||
| FROM python:3.6-slim-stretch as build | FROM docker.beryju.org/passbook/base:latest | ||||||
|  |  | ||||||
| COPY ./passbook/ /app/passbook | COPY ./passbook/ /app/passbook | ||||||
| COPY ./manage.py /app/ | COPY ./manage.py /app/ | ||||||
| COPY ./requirements.txt /app/ | COPY ./docker/uwsgi.ini /app/ | ||||||
|  |  | ||||||
| WORKDIR /app/ | WORKDIR /app/ | ||||||
|  |  | ||||||
| RUN mkdir /app/static/ && \ |  | ||||||
|     pip install -r requirements.txt && \ |  | ||||||
|     pip install psycopg2 && \ |  | ||||||
|     ./manage.py collectstatic --no-input |  | ||||||
|  |  | ||||||
| FROM python:3.6-slim-stretch |  | ||||||
|  |  | ||||||
| COPY ./passbook/ /app/passbook |  | ||||||
| COPY ./manage.py /app/ |  | ||||||
| COPY ./requirements.txt /app/ |  | ||||||
| COPY --from=build /app/static /app/static/ |  | ||||||
|  |  | ||||||
| WORKDIR /app/ |  | ||||||
|  |  | ||||||
| RUN pip install -r requirements.txt && \ |  | ||||||
|     pip install psycopg2 && \ |  | ||||||
|     adduser --system --home /app/ passbook && \ |  | ||||||
|     chown -R passbook /app/ |  | ||||||
|  |  | ||||||
| USER passbook | USER passbook | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @ -1,6 +1,6 @@ | |||||||
| MIT License | MIT License | ||||||
|  |  | ||||||
| Copyright (c) 2018 BeryJu.org | Copyright (c) 2019 BeryJu.org | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
| of this software and associated documentation files (the "Software"), to deal | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | |||||||
							
								
								
									
										59
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | [[source]] | ||||||
|  | name = "pypi" | ||||||
|  | url = "https://pypi.org/simple" | ||||||
|  | verify_ssl = true | ||||||
|  |  | ||||||
|  | [packages] | ||||||
|  | boto3 = "*" | ||||||
|  | celery = "*" | ||||||
|  | defusedxml = "*" | ||||||
|  | django = "*" | ||||||
|  | django-cors-middleware = "*" | ||||||
|  | django-dbbackup = "*" | ||||||
|  | django-filter = "*" | ||||||
|  | django-guardian = "*" | ||||||
|  | django-ipware = "*" | ||||||
|  | django-model-utils = "*" | ||||||
|  | django-oauth-toolkit = "*" | ||||||
|  | django-oidc-provider = "*" | ||||||
|  | django-otp = "*" | ||||||
|  | django-prometheus = "*" | ||||||
|  | django-recaptcha = "*" | ||||||
|  | django-redis = "*" | ||||||
|  | django-rest-framework = "*" | ||||||
|  | django-storages = "*" | ||||||
|  | djangorestframework-guardian = "*" | ||||||
|  | drf-yasg = "*" | ||||||
|  | kombu = "==4.5.0" | ||||||
|  | ldap3 = "*" | ||||||
|  | lxml = "*" | ||||||
|  | oauthlib = "*" | ||||||
|  | packaging = "*" | ||||||
|  | psycopg2-binary = "*" | ||||||
|  | pycryptodome = "*" | ||||||
|  | pyuwsgi = "*" | ||||||
|  | pyyaml = "*" | ||||||
|  | qrcode = "*" | ||||||
|  | requests-oauthlib = "*" | ||||||
|  | sentry-sdk = "*" | ||||||
|  | service_identity = "*" | ||||||
|  | signxml = "*" | ||||||
|  | structlog = "*" | ||||||
|  | swagger-spec-validator = "*" | ||||||
|  | urllib3 = {extras = ["secure"],version = "*"} | ||||||
|  |  | ||||||
|  | [requires] | ||||||
|  | python_version = "3.7" | ||||||
|  |  | ||||||
|  | [dev-packages] | ||||||
|  | autopep8 = "*" | ||||||
|  | bandit = "*" | ||||||
|  | bumpversion = "*" | ||||||
|  | colorama = "*" | ||||||
|  | coverage = "*" | ||||||
|  | django-debug-toolbar = "*" | ||||||
|  | isort = "*" | ||||||
|  | prospector = "*" | ||||||
|  | pylint = "==2.3.1" | ||||||
|  | pylint-django = "*" | ||||||
|  | unittest-xml-reporting = "*" | ||||||
							
								
								
									
										1211
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										1211
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | # passbook | ||||||
|  |  | ||||||
|  | ## Quick instance | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | export PASSBOOK_DOMAIN=domain.tld | ||||||
|  | docker-compose pull | ||||||
|  | docker-compose up -d | ||||||
|  | docker-compose exec server ./manage.py migrate | ||||||
|  | docker-compose exec server ./manage.py createsuperuser | ||||||
|  | ``` | ||||||
| @ -1,27 +0,0 @@ | |||||||
| # Global Variables |  | ||||||
| before_script: |  | ||||||
|   - cd allauth/ |  | ||||||
|   - "python3 -m pip install -U virtualenv" |  | ||||||
|   - "virtualenv env" |  | ||||||
|   - "source env/bin/activate" |  | ||||||
|   - "pip3 install -U -r requirements-dev.txt" |  | ||||||
| stages: |  | ||||||
|   - test-allauth |  | ||||||
| image: python:3.6 |  | ||||||
|  |  | ||||||
| isort: |  | ||||||
|   script: |  | ||||||
|     - isort -c -sg env |  | ||||||
|   stage: test-allauth |  | ||||||
| prospector: |  | ||||||
|   script: |  | ||||||
|     - prospector |  | ||||||
|   stage: test-allauth |  | ||||||
| pylint: |  | ||||||
|   script: |  | ||||||
|     - pylint passbook |  | ||||||
|   stage: test-allauth |  | ||||||
| bandit: |  | ||||||
|   script: |  | ||||||
|     - bandit -r allauth_passbook |  | ||||||
|   stage: test-allauth |  | ||||||
| @ -1,35 +0,0 @@ | |||||||
| """passbook provider""" |  | ||||||
| from allauth.socialaccount.providers.base import ProviderAccount |  | ||||||
| from allauth.socialaccount.providers.oauth2.provider import OAuth2Provider |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PassbookAccount(ProviderAccount): |  | ||||||
|     """passbook account""" |  | ||||||
|  |  | ||||||
|     def to_str(self): |  | ||||||
|         dflt = super().to_str() |  | ||||||
|         return self.account.extra_data.get('username', dflt) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PassbookProvider(OAuth2Provider): |  | ||||||
|     """passbook provider""" |  | ||||||
|  |  | ||||||
|     id = 'passbook' |  | ||||||
|     name = 'passbook' |  | ||||||
|     account_class = PassbookAccount |  | ||||||
|  |  | ||||||
|     def extract_uid(self, data): |  | ||||||
|         return str(data['sub']) |  | ||||||
|  |  | ||||||
|     def extract_common_fields(self, data): |  | ||||||
|         return { |  | ||||||
|             'email': data.get('email'), |  | ||||||
|             'username': data.get('preferred_username'), |  | ||||||
|             'name': data.get('name'), |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def get_default_scope(self): |  | ||||||
|         return ['openid:userinfo'] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| provider_classes = [PassbookProvider] # noqa |  | ||||||
| @ -1,5 +0,0 @@ | |||||||
| """passbook provider""" |  | ||||||
| from allauth.socialaccount.providers.oauth2.urls import default_urlpatterns |  | ||||||
| from allauth_passbook.provider import PassbookProvider |  | ||||||
|  |  | ||||||
| urlpatterns = default_urlpatterns(PassbookProvider) |  | ||||||
| @ -1,37 +0,0 @@ | |||||||
| """passbook adapter""" |  | ||||||
| import requests |  | ||||||
|  |  | ||||||
| from allauth.socialaccount import app_settings |  | ||||||
| from allauth.socialaccount.providers.oauth2.views import (OAuth2Adapter, |  | ||||||
|                                                           OAuth2CallbackView, |  | ||||||
|                                                           OAuth2LoginView) |  | ||||||
| from allauth_passbook.provider import PassbookProvider |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PassbookOAuth2Adapter(OAuth2Adapter): |  | ||||||
|     """passbook OAuth2 Adapter""" |  | ||||||
|     provider_id = PassbookProvider.id |  | ||||||
|     # pylint: disable=no-member |  | ||||||
|     settings = app_settings.PROVIDERS.get(provider_id, {}) # noqa |  | ||||||
|     provider_base_url = settings.get("PASSBOOK_URL", 'https://id.beryju.org') |  | ||||||
|  |  | ||||||
|     access_token_url = '{0}/application/oauth/token/'.format(provider_base_url) |  | ||||||
|     authorize_url = '{0}/application/oauth/authorize/'.format(provider_base_url) |  | ||||||
|     profile_url = '{0}/api/v1/openid/'.format( |  | ||||||
|         provider_base_url) |  | ||||||
|  |  | ||||||
|     def complete_login(self, request, app, access_token, **kwargs): |  | ||||||
|         headers = { |  | ||||||
|             'Authorization': 'Bearer {0}'.format(access_token.token), |  | ||||||
|             'Content-Type': 'application/json', |  | ||||||
|         } |  | ||||||
|         extra_data = requests.get(self.profile_url, headers=headers) |  | ||||||
|  |  | ||||||
|         return self.get_provider().sociallogin_from_response( |  | ||||||
|             request, |  | ||||||
|             extra_data.json() |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| oauth2_login = OAuth2LoginView.adapter_view(PassbookOAuth2Adapter) # noqa |  | ||||||
| oauth2_callback = OAuth2CallbackView.adapter_view(PassbookOAuth2Adapter) # noqa |  | ||||||
| @ -1 +0,0 @@ | |||||||
| django-allauth |  | ||||||
| @ -1,33 +0,0 @@ | |||||||
| """passbook allauth setup.py""" |  | ||||||
| from setuptools import setup |  | ||||||
|  |  | ||||||
| setup( |  | ||||||
|     name='django-allauth-passbook', |  | ||||||
|     version='1.0.0', |  | ||||||
|     description='passbook support for django-allauth', |  | ||||||
|     # long_description='\n'.join(read_simple('docs/index.md')[2:]), |  | ||||||
|     long_description_content_type='text/markdown', |  | ||||||
|     author='BeryJu.org', |  | ||||||
|     author_email='hello@beryju.org', |  | ||||||
|     packages=['allauth_passbook'], |  | ||||||
|     include_package_data=True, |  | ||||||
|     install_requires=['django-allauth'], |  | ||||||
|     keywords='django allauth passbook', |  | ||||||
|     license='MIT', |  | ||||||
|     classifiers=[ |  | ||||||
|         'Intended Audience :: Developers', |  | ||||||
|         'Topic :: Software Development :: Libraries :: Python Modules', |  | ||||||
|         'Environment :: Web Environment', |  | ||||||
|         'Topic :: Internet', |  | ||||||
|         'License :: OSI Approved :: MIT License', |  | ||||||
|         'Operating System :: OS Independent', |  | ||||||
|         'Programming Language :: Python', |  | ||||||
|         'Programming Language :: Python :: 3.4', |  | ||||||
|         'Programming Language :: Python :: 3.5', |  | ||||||
|         'Programming Language :: Python :: 3.6', |  | ||||||
|         'Framework :: Django', |  | ||||||
|         'Framework :: Django :: 1.11', |  | ||||||
|         'Framework :: Django :: 2.0', |  | ||||||
|         'Framework :: Django :: 2.1', |  | ||||||
|     ], |  | ||||||
| ) |  | ||||||
							
								
								
									
										23
									
								
								base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | FROM python:3.7-slim-buster as locker | ||||||
|  |  | ||||||
|  | COPY ./Pipfile /app/ | ||||||
|  | COPY ./Pipfile.lock /app/ | ||||||
|  |  | ||||||
|  | WORKDIR /app/ | ||||||
|  |  | ||||||
|  | RUN pip install pipenv && \ | ||||||
|  |     pipenv lock -r > requirements.txt && \ | ||||||
|  |     pipenv lock -rd > requirements-dev.txt | ||||||
|  |  | ||||||
|  | FROM python:3.7-slim-buster | ||||||
|  |  | ||||||
|  | COPY --from=locker /app/requirements.txt /app/ | ||||||
|  | COPY --from=locker /app/requirements-dev.txt /app/ | ||||||
|  |  | ||||||
|  | WORKDIR /app/ | ||||||
|  |  | ||||||
|  | RUN apt-get update && \ | ||||||
|  |     apt-get install -y --no-install-recommends postgresql-client-11 && \ | ||||||
|  |     rm -rf /var/lib/apt/ && \ | ||||||
|  |     pip install -r requirements.txt  --no-cache-dir && \ | ||||||
|  |     adduser --system --no-create-home --uid 1000 --group --home /app passbook | ||||||
							
								
								
									
										3
									
								
								dev.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								dev.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | FROM docker.beryju.org/passbook/base:latest | ||||||
|  |  | ||||||
|  | RUN pip install -r /app/requirements-dev.txt  --no-cache-dir | ||||||
							
								
								
									
										90
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,90 @@ | |||||||
|  | --- | ||||||
|  | version: '3.2' | ||||||
|  |  | ||||||
|  | services: | ||||||
|  |   postgresql: | ||||||
|  |     image: postgres | ||||||
|  |     volumes: | ||||||
|  |       - database:/var/lib/postgresql/data | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     environment: | ||||||
|  |       - POSTGRES_PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |       - POSTGRES_USER=passbook | ||||||
|  |       - POSTGRES_DB=passbook | ||||||
|  |     labels: | ||||||
|  |       - traefik.enable=false | ||||||
|  |   redis: | ||||||
|  |     image: redis | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.enable=false | ||||||
|  |   server: | ||||||
|  |     build: | ||||||
|  |       context: . | ||||||
|  |     image: docker.beryju.org/passbook/server:${SERVER_TAG:-latest} | ||||||
|  |     command: | ||||||
|  |       - uwsgi | ||||||
|  |       - uwsgi.ini | ||||||
|  |     environment: | ||||||
|  |       - PASSBOOK_DOMAIN=${PASSBOOK_DOMAIN} | ||||||
|  |       - PASSBOOK_REDIS__HOST=redis | ||||||
|  |       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||||
|  |       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |     ports: | ||||||
|  |       - 8000 | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.port=8000 | ||||||
|  |       - traefik.docker.network=internal | ||||||
|  |       - traefik.frontend.rule=PathPrefix:/ | ||||||
|  |   worker: | ||||||
|  |     image: docker.beryju.org/passbook/server:${SERVER_TAG:-latest} | ||||||
|  |     command: | ||||||
|  |       - celery | ||||||
|  |       - worker | ||||||
|  |       - --autoscale=10,3 | ||||||
|  |       - -E | ||||||
|  |       - -B | ||||||
|  |       - -A=passbook.root.celery | ||||||
|  |       - -s=/tmp/celerybeat-schedule | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.enable=false | ||||||
|  |     environment: | ||||||
|  |       - PASSBOOK_DOMAIN=${PASSBOOK_DOMAIN} | ||||||
|  |       - PASSBOOK_REDIS__HOST=redis | ||||||
|  |       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||||
|  |       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |   static: | ||||||
|  |     build: | ||||||
|  |       context: . | ||||||
|  |       dockerfile: static.Dockerfile | ||||||
|  |     image: docker.beryju.org/passbook/static:latest | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.frontend.rule=PathPrefix:/static, /robots.txt | ||||||
|  |       - traefik.port=80 | ||||||
|  |       - traefik.docker.network=internal | ||||||
|  |   traefik: | ||||||
|  |     image: traefik:1.7 | ||||||
|  |     command: --api --docker | ||||||
|  |     volumes: | ||||||
|  |       - /var/run/docker.sock:/var/run/docker.sock:ro | ||||||
|  |     ports: | ||||||
|  |       - "0.0.0.0:80:80" | ||||||
|  |       - "0.0.0.0:443:443" | ||||||
|  |       - "0.0.0.0:8080:8080" | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |  | ||||||
|  | volumes: | ||||||
|  |   database: | ||||||
|  |     driver: local | ||||||
|  |  | ||||||
|  | networks: | ||||||
|  |   internal: {} | ||||||
							
								
								
									
										66
									
								
								docker/nginx.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								docker/nginx.conf
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | |||||||
|  | user  nginx; | ||||||
|  | worker_processes  1; | ||||||
|  |  | ||||||
|  | error_log  stderr warn; | ||||||
|  | pid        /var/run/nginx.pid; | ||||||
|  |  | ||||||
|  | events { | ||||||
|  |     worker_connections  1024; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | http { | ||||||
|  |     include       /etc/nginx/mime.types; | ||||||
|  |     default_type  application/octet-stream; | ||||||
|  |  | ||||||
|  |     log_format json_combined escape=json | ||||||
|  |         '{' | ||||||
|  |             '"time_local":"$time_local",' | ||||||
|  |             '"remote_addr":"$remote_addr",' | ||||||
|  |             '"remote_user":"$remote_user",' | ||||||
|  |             '"request":"$request",' | ||||||
|  |             '"status": "$status",' | ||||||
|  |             '"body_bytes_sent":"$body_bytes_sent",' | ||||||
|  |             '"request_time":"$request_time",' | ||||||
|  |             '"http_referrer":"$http_referer",' | ||||||
|  |             '"http_user_agent":"$http_user_agent"' | ||||||
|  |         '}'; | ||||||
|  |  | ||||||
|  |     access_log /dev/stdout json_combined; | ||||||
|  |  | ||||||
|  |     sendfile        on; | ||||||
|  |     tcp_nopush     on; | ||||||
|  |  | ||||||
|  |     keepalive_timeout  65; | ||||||
|  |  | ||||||
|  |     server { | ||||||
|  |  | ||||||
|  |         server_name _; | ||||||
|  |  | ||||||
|  |         gzip on; | ||||||
|  |         gzip_types application/javascript image/* text/css; | ||||||
|  |         gunzip on; | ||||||
|  |         add_header X-passbook-Version 0.7.2-beta; | ||||||
|  |         add_header Vary X-passbook-Version; | ||||||
|  |         root /data/; | ||||||
|  |  | ||||||
|  |         location /_/healthz { | ||||||
|  |             return 204; | ||||||
|  |         } | ||||||
|  |         location ~* \.(jpg|jpeg|png|gif|ico)$ { | ||||||
|  |             expires 30d; | ||||||
|  |         } | ||||||
|  |         location ~* \.(css|js)$ { | ||||||
|  |             expires 7d; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     server { | ||||||
|  |  | ||||||
|  |         listen 8080; | ||||||
|  |  | ||||||
|  |         location = /stub_status { | ||||||
|  |             stub_status; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								docker/uwsgi.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								docker/uwsgi.ini
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | [uwsgi] | ||||||
|  | http = 0.0.0.0:8000 | ||||||
|  | wsgi-file = passbook/root/wsgi.py | ||||||
|  | processes = 2 | ||||||
|  | master = true | ||||||
|  | threads = 2 | ||||||
|  | enable-threads = true | ||||||
|  | uid = passbook | ||||||
|  | gid = passbook | ||||||
|  | disable-logging=True | ||||||
							
								
								
									
										10
									
								
								hack/prometheus/grafana.helm.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								hack/prometheus/grafana.helm.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | ingress: | ||||||
|  |   enabled: true | ||||||
|  |   hosts: | ||||||
|  |     - some.address.tld | ||||||
|  |  | ||||||
|  | grafana.ini: | ||||||
|  |   auth.anonymous: | ||||||
|  |     enabled: true | ||||||
|  |     org_name: Main Org. | ||||||
|  |     org_role: Viewer | ||||||
							
								
								
									
										63
									
								
								hack/prometheus/instance.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								hack/prometheus/instance.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | --- | ||||||
|  | apiVersion: v1 | ||||||
|  | kind: ServiceAccount | ||||||
|  | metadata: | ||||||
|  |     name: prometheus | ||||||
|  | --- | ||||||
|  | apiVersion: rbac.authorization.k8s.io/v1beta1 | ||||||
|  | kind: ClusterRole | ||||||
|  | metadata: | ||||||
|  |     name: prometheus | ||||||
|  | rules: | ||||||
|  |     - apiGroups: [""] | ||||||
|  |       resources: | ||||||
|  |           - nodes | ||||||
|  |           - services | ||||||
|  |           - endpoints | ||||||
|  |           - pods | ||||||
|  |       verbs: ["get", "list", "watch"] | ||||||
|  |     - apiGroups: [""] | ||||||
|  |       resources: | ||||||
|  |           - configmaps | ||||||
|  |       verbs: ["get"] | ||||||
|  |     - nonResourceURLs: ["/metrics"] | ||||||
|  |       verbs: ["get"] | ||||||
|  | --- | ||||||
|  | apiVersion: rbac.authorization.k8s.io/v1beta1 | ||||||
|  | kind: ClusterRoleBinding | ||||||
|  | metadata: | ||||||
|  |     name: prometheus | ||||||
|  | roleRef: | ||||||
|  |     apiGroup: rbac.authorization.k8s.io | ||||||
|  |     kind: ClusterRole | ||||||
|  |     name: prometheus | ||||||
|  | subjects: | ||||||
|  |     - kind: ServiceAccount | ||||||
|  |       name: prometheus | ||||||
|  |       namespace: prod-passbook-ng | ||||||
|  | --- | ||||||
|  | apiVersion: monitoring.coreos.com/v1 | ||||||
|  | kind: Prometheus | ||||||
|  | metadata: | ||||||
|  |     name: prometheus | ||||||
|  | spec: | ||||||
|  |     serviceAccountName: prometheus | ||||||
|  |     serviceMonitorSelector: | ||||||
|  |         matchLabels: | ||||||
|  |             app.kubernetes.io/name: passbook | ||||||
|  |     enableAdminAPI: false | ||||||
|  |     ruleSelector: | ||||||
|  |         matchLabels: | ||||||
|  |             app.kubernetes.io/name: passbook | ||||||
|  |     storage: | ||||||
|  |         volumeClaimTemplate: | ||||||
|  |             metadata: | ||||||
|  |                 labels: | ||||||
|  |                     prometheus: k8s | ||||||
|  |                 name: prometheus-storage | ||||||
|  |             spec: | ||||||
|  |                 accessModes: | ||||||
|  |                     - ReadWriteOnce | ||||||
|  |                 resources: | ||||||
|  |                     requests: | ||||||
|  |                         storage: 15Gi | ||||||
							
								
								
									
										11
									
								
								hack/up.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										11
									
								
								hack/up.sh
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | #!/bin/bash -x | ||||||
|  |  | ||||||
|  | # macos specific setting, for some reason | ||||||
|  | export OBJC_DISABLE_INITIALIZE_FORK_SAFETY=YES | ||||||
|  | export DEBUG=false | ||||||
|  |  | ||||||
|  | export POSTGRES_USER=postgres | ||||||
|  |  | ||||||
|  | # ./manage.py generate_swagger > storhappy-ui/swagger.json | ||||||
|  |  | ||||||
|  | uwsgi docker/uwsgi.ini | ||||||
							
								
								
									
										9
									
								
								helm/passbook/Chart.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								helm/passbook/Chart.lock
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | dependencies: | ||||||
|  | - name: postgresql | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 6.5.8 | ||||||
|  | - name: redis | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 9.5.1 | ||||||
|  | digest: sha256:f18b5dc8d0be13d584407405c60d10b6b84d25f7fa8aaa3dd0e5385c38f5c516 | ||||||
|  | generated: "2019-11-07T10:23:07.259176+01:00" | ||||||
| @ -1,6 +1,6 @@ | |||||||
| apiVersion: v1 | apiVersion: v1 | ||||||
| appVersion: "0.0.8-alpha" | appVersion: "0.7.2-beta" | ||||||
| description: A Helm chart for passbook. | description: A Helm chart for passbook. | ||||||
| name: passbook | name: passbook | ||||||
| version: 1.0.0 | version: "0.7.2-beta" | ||||||
| icon: https://passbook.beryju.org/images/logo.png | icon: https://git.beryju.org/uploads/-/system/project/avatar/108/logo.png | ||||||
|  | |||||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @ -1,9 +1,9 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis |  | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |  | ||||||
|   version: 5.1.0 |  | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|   version: 3.10.1 |   version: 6.5.8 | ||||||
| digest: sha256:04bd136761f070e94a2ff32ff48ff87f5e07fbd451e5fd7f65551e3bd4680e5e | - name: redis | ||||||
| generated: 2019-02-08T12:08:49.090666+01:00 |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 9.5.1 | ||||||
|  | digest: sha256:476834fb82f66bc7242c4a5e7343d0a859d8307cb301256beb0eb749983014e4 | ||||||
|  | generated: "2019-11-07T10:21:30.902415+01:00" | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis |  | ||||||
|   version: 5.1.0 |  | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |  | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   version: 3.10.1 |   version: 6.5.8 | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | - name: redis | ||||||
|  |   version: 9.5.1 | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								helm/passbook/templates/configmap.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								helm/passbook/templates/configmap.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | apiVersion: v1 | ||||||
|  | kind: ConfigMap | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-config | ||||||
|  | data: | ||||||
|  |   config.yml: | | ||||||
|  |     postgresql: | ||||||
|  |       host: "{{ .Release.Name }}-postgresql" | ||||||
|  |       name: "{{ .Values.postgresql.postgresqlDatabase }}" | ||||||
|  |       user: postgres | ||||||
|  |     redis: | ||||||
|  |       host: "{{ .Release.Name }}-redis-master" | ||||||
|  |       cache_db: 0 | ||||||
|  |       message_queue_db: 1 | ||||||
|  |     error_report_enabled: {{ .Values.config.error_reporting }} | ||||||
|  |     domain: ".{{ index .Values.ingress.hosts 0 }}" | ||||||
| @ -1,6 +1,5 @@ | |||||||
| {{- if .Values.ingress.enabled -}} | {{- if .Values.ingress.enabled -}} | ||||||
| {{- $fullName := include "passbook.fullname" . -}} | {{- $fullName := include "passbook.fullname" . -}} | ||||||
| {{- $ingressPath := .Values.ingress.path -}} |  | ||||||
| apiVersion: extensions/v1beta1 | apiVersion: extensions/v1beta1 | ||||||
| kind: Ingress | kind: Ingress | ||||||
| metadata: | metadata: | ||||||
| @ -30,9 +29,17 @@ spec: | |||||||
|     - host: {{ . | quote }} |     - host: {{ . | quote }} | ||||||
|       http: |       http: | ||||||
|         paths: |         paths: | ||||||
|           - path: {{ $ingressPath }} |           - path: / | ||||||
|             backend: |             backend: | ||||||
|               serviceName: {{ $fullName }} |               serviceName: {{ $fullName }}-web | ||||||
|  |               servicePort: http | ||||||
|  |           - path: /static/ | ||||||
|  |             backend: | ||||||
|  |               serviceName: {{ $fullName }}-static | ||||||
|  |               servicePort: http | ||||||
|  |           - path: /robots.txt | ||||||
|  |             backend: | ||||||
|  |               serviceName: {{ $fullName }}-static | ||||||
|               servicePort: http |               servicePort: http | ||||||
|   {{- end }} |   {{- end }} | ||||||
| {{- end }} | {{- end }} | ||||||
| @ -1,138 +0,0 @@ | |||||||
| apiVersion: v1 |  | ||||||
| kind: ConfigMap |  | ||||||
| metadata: |  | ||||||
|   name: {{ include "passbook.fullname" . }}-config |  | ||||||
| data: |  | ||||||
|   config.yml: | |  | ||||||
|     # Env for Docker images |  | ||||||
|     databases: |  | ||||||
|       default: |  | ||||||
|         engine: django.db.backends.postgresql |  | ||||||
|         name: {{ .Values.postgresql.postgresqlDatabase }} |  | ||||||
|         user: postgres |  | ||||||
|         password: {{ .Values.postgresql.postgresqlPassword }} |  | ||||||
|         host: {{ .Release.Name }}-postgresql |  | ||||||
|         port: '' |  | ||||||
|     log: |  | ||||||
|       level: |  | ||||||
|         console: DEBUG |  | ||||||
|         file: DEBUG |  | ||||||
|       file: /dev/null |  | ||||||
|       syslog: |  | ||||||
|         host: 127.0.0.1 |  | ||||||
|         port: 514 |  | ||||||
|     email: |  | ||||||
|       host: localhost |  | ||||||
|       port: 25 |  | ||||||
|       user: '' |  | ||||||
|       password: '' |  | ||||||
|       use_tls: false |  | ||||||
|       use_ssl: false |  | ||||||
|       from: passbook <passbook@domain.tld> |  | ||||||
|     web: |  | ||||||
|       listen: 0.0.0.0 |  | ||||||
|       port: 8000 |  | ||||||
|       threads: 30 |  | ||||||
|     debug: false |  | ||||||
|     secure_proxy_header: |  | ||||||
|       HTTP_X_FORWARDED_PROTO: https |  | ||||||
|     redis: {{ .Release.Name }}-redis |  | ||||||
|     # Error reporting, sends stacktrace to sentry.services.beryju.org |  | ||||||
|     error_report_enabled: {{ .Values.config.error_reporting }} |  | ||||||
|  |  | ||||||
|     {{- if .Values.config.secret_key }} |  | ||||||
|     secret_key: {{ .Values.config.secret_key }} |  | ||||||
|     {{- else }} |  | ||||||
|     secret_key: {{ randAlphaNum 50 }} |  | ||||||
|     {{- end }} |  | ||||||
|  |  | ||||||
|     domains: |  | ||||||
|         {{- range .Values.ingress.hosts }} |  | ||||||
|         - {{ . | quote }} |  | ||||||
|         {{- end }} |  | ||||||
|  |  | ||||||
|     passbook: |  | ||||||
|       sign_up: |  | ||||||
|         # Enables signup, created users are stored in internal Database and created in LDAP if ldap.create_users is true |  | ||||||
|         enabled: true |  | ||||||
|       password_reset: |  | ||||||
|         # Enable password reset, passwords are reset in internal Database and in LDAP if ldap.reset_password is true |  | ||||||
|         enabled: true |  | ||||||
|         # Verification the user has to provide in order to be able to reset passwords. Can be any combination of `email`, `2fa`, `security_questions` |  | ||||||
|         verification: |  | ||||||
|           - email |  | ||||||
|       # Text used in title, on login page and multiple other places |  | ||||||
|       branding: passbook |  | ||||||
|       login: |  | ||||||
|         # Override URL used for logo |  | ||||||
|         logo_url: null |  | ||||||
|         # Override URL used for Background on Login page |  | ||||||
|         bg_url: null |  | ||||||
|         # Optionally add a subtext, placed below logo on the login page |  | ||||||
|         subtext: null |  | ||||||
|       footer: |  | ||||||
|         links: |  | ||||||
|           # Optionally add links to the footer on the login page |  | ||||||
|           #  - name: test |  | ||||||
|           #    href: https://test |  | ||||||
|       # Specify which fields can be used to authenticate. Can be any combination of `username` and `email` |  | ||||||
|       uid_fields: |  | ||||||
|         - username |  | ||||||
|         - email |  | ||||||
|       session: |  | ||||||
|         remember_age: 2592000 # 60 * 60 * 24 * 30, one month |  | ||||||
|     # Provider-specific settings |  | ||||||
|     ldap: |  | ||||||
|       # # Completely enable or disable LDAP provider |  | ||||||
|       # enabled: false |  | ||||||
|       # # AD Domain, used to generate `userPrincipalName` |  | ||||||
|       # domain: corp.contoso.com |  | ||||||
|       # # Base DN in which passbook should look for users |  | ||||||
|       # base_dn: dn=corp,dn=contoso,dn=com |  | ||||||
|       # # LDAP field which is used to set the django username |  | ||||||
|       # username_field: sAMAccountName |  | ||||||
|       # # LDAP server to connect to, can be set to `<domain_name>` |  | ||||||
|       # server: |  | ||||||
|       #   name: corp.contoso.com |  | ||||||
|       #   use_tls: false |  | ||||||
|       # # Bind credentials, used for account creation |  | ||||||
|       # bind: |  | ||||||
|       #   username: Administraotr@corp.contoso.com |  | ||||||
|       #   password: VerySecurePassword! |  | ||||||
|       # Which field from `uid_fields` maps to which LDAP Attribute |  | ||||||
|       login_field_map: |  | ||||||
|         username: sAMAccountName |  | ||||||
|         email: mail # or userPrincipalName |  | ||||||
|       user_attribute_map: |  | ||||||
|         active_directory: |  | ||||||
|           sAMAccountName: username |  | ||||||
|           mail: email |  | ||||||
|           given_name: first_name |  | ||||||
|           name: last_name |  | ||||||
|       # # Create new users in LDAP upon sign-up |  | ||||||
|       # create_users: true |  | ||||||
|       # # Reset LDAP password when user reset their password |  | ||||||
|       # reset_password: true |  | ||||||
|     oauth_client: |  | ||||||
|       # List of python packages with sources types to load. |  | ||||||
|       types: |  | ||||||
|         - passbook.oauth_client.source_types.discord |  | ||||||
|         - passbook.oauth_client.source_types.facebook |  | ||||||
|         - passbook.oauth_client.source_types.github |  | ||||||
|         - passbook.oauth_client.source_types.google |  | ||||||
|         - passbook.oauth_client.source_types.reddit |  | ||||||
|         - passbook.oauth_client.source_types.supervisr |  | ||||||
|         - passbook.oauth_client.source_types.twitter |  | ||||||
|     saml_idp: |  | ||||||
|       signing: true |  | ||||||
|       autosubmit: false |  | ||||||
|       issuer: passbook |  | ||||||
|       assertion_valid_for: 86400 |  | ||||||
|       # List of python packages with provider types to load. |  | ||||||
|       types: |  | ||||||
|         - passbook.saml_idp.processors.generic |  | ||||||
|         - passbook.saml_idp.processors.gitlab |  | ||||||
|         - passbook.saml_idp.processors.nextcloud |  | ||||||
|         - passbook.saml_idp.processors.salesforce |  | ||||||
|         - passbook.saml_idp.processors.shibboleth |  | ||||||
|         - passbook.saml_idp.processors.wordpress_orange |  | ||||||
| @ -1,66 +0,0 @@ | |||||||
| apiVersion: apps/v1beta2 |  | ||||||
| kind: Deployment |  | ||||||
| metadata: |  | ||||||
|   name: {{ include "passbook.fullname" . }}-web |  | ||||||
|   labels: |  | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |  | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     app.kubernetes.io/managed-by: {{ .Release.Service }} |  | ||||||
| spec: |  | ||||||
|   replicas: {{ .Values.replicaCount }} |  | ||||||
|   selector: |  | ||||||
|     matchLabels: |  | ||||||
|       app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|       app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|   template: |  | ||||||
|     metadata: |  | ||||||
|       labels: |  | ||||||
|         app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|         app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     spec: |  | ||||||
|       volumes: |  | ||||||
|         - name: config-volume |  | ||||||
|           configMap: |  | ||||||
|             name: {{ include "passbook.fullname" . }}-config |  | ||||||
|       containers: |  | ||||||
|         - name: {{ .Chart.Name }} |  | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |  | ||||||
|           imagePullPolicy: IfNotPresent |  | ||||||
|           command: ["/bin/sh","-c"] |  | ||||||
|           args: ["./manage.py migrate && ./manage.py web"] |  | ||||||
|           ports: |  | ||||||
|             - name: http |  | ||||||
|               containerPort: 8000 |  | ||||||
|               protocol: TCP |  | ||||||
|           volumeMounts: |  | ||||||
|             - mountPath: /etc/passbook |  | ||||||
|               name: config-volume |  | ||||||
|           livenessProbe: |  | ||||||
|             httpGet: |  | ||||||
|               path: / |  | ||||||
|               port: http |  | ||||||
|               httpHeaders: |  | ||||||
|                 - name: Host |  | ||||||
|                   value: kubernetes-healthcheck-host |  | ||||||
|           readinessProbe: |  | ||||||
|             httpGet: |  | ||||||
|               path: / |  | ||||||
|               port: http |  | ||||||
|               httpHeaders: |  | ||||||
|                 - name: Host |  | ||||||
|                   value: kubernetes-healthcheck-host |  | ||||||
|           resources: |  | ||||||
| {{ toYaml .Values.resources | indent 12 }} |  | ||||||
|     {{- with .Values.nodeSelector }} |  | ||||||
|       nodeSelector: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.affinity }} |  | ||||||
|       affinity: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.tolerations }} |  | ||||||
|       tolerations: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
| @ -1,51 +0,0 @@ | |||||||
| apiVersion: apps/v1beta2 |  | ||||||
| kind: Deployment |  | ||||||
| metadata: |  | ||||||
|   name: {{ include "passbook.fullname" . }}-worker |  | ||||||
|   labels: |  | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |  | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     app.kubernetes.io/managed-by: {{ .Release.Service }} |  | ||||||
| spec: |  | ||||||
|   replicas: {{ .Values.replicaCount }} |  | ||||||
|   selector: |  | ||||||
|     matchLabels: |  | ||||||
|       app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|       app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|   template: |  | ||||||
|     metadata: |  | ||||||
|       labels: |  | ||||||
|         app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|         app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     spec: |  | ||||||
|       volumes: |  | ||||||
|         - name: config-volume |  | ||||||
|           configMap: |  | ||||||
|             name: {{ include "passbook.fullname" . }}-config |  | ||||||
|       containers: |  | ||||||
|         - name: {{ .Chart.Name }} |  | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |  | ||||||
|           imagePullPolicy: IfNotPresent |  | ||||||
|           command: ["./manage.py", "worker"] |  | ||||||
|           ports: |  | ||||||
|             - name: http |  | ||||||
|               containerPort: 8000 |  | ||||||
|               protocol: TCP |  | ||||||
|           volumeMounts: |  | ||||||
|             - mountPath: /etc/passbook |  | ||||||
|               name: config-volume |  | ||||||
|           resources: |  | ||||||
| {{ toYaml .Values.resources | indent 12 }} |  | ||||||
|     {{- with .Values.nodeSelector }} |  | ||||||
|       nodeSelector: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.affinity }} |  | ||||||
|       affinity: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.tolerations }} |  | ||||||
|       tolerations: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
							
								
								
									
										124
									
								
								helm/passbook/templates/prom-rules.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								helm/passbook/templates/prom-rules.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,124 @@ | |||||||
|  | {{- if .Values.monitoring.enabled -}} | ||||||
|  | --- | ||||||
|  | apiVersion: monitoring.coreos.com/v1 | ||||||
|  | kind: PrometheusRule | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-static-rules | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  | spec: | ||||||
|  |   groups: | ||||||
|  |   - name: Aggregate request counters | ||||||
|  |     rules: | ||||||
|  |       - record: job:django_http_requests_before_middlewares_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_before_middlewares_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_requests_unknown_latency_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_unknown_latency_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_ajax_requests_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_ajax_requests_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_responses_before_middlewares_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_before_middlewares_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_requests_unknown_latency_including_middlewares_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_unknown_latency_including_middlewares_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_requests_body_total_bytes:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_body_total_bytes[30s])) by (job) | ||||||
|  |       - record: job:django_http_responses_streaming_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_streaming_total[30s])) by (job) | ||||||
|  |       - record: job:django_http_responses_body_total_bytes:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_body_total_bytes[30s])) by (job) | ||||||
|  |       - record: job:django_http_requests_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_total_by_method[30s])) by (job) | ||||||
|  |       - record: job:django_http_requests_total_by_method:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_total_by_method[30s])) by (job,method) | ||||||
|  |       - record: job:django_http_requests_total_by_transport:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_total_by_transport[30s])) by (job,transport) | ||||||
|  |       - record: job:django_http_requests_total_by_view:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_total_by_view_transport_method[30s])) by (job,view) | ||||||
|  |       - record: job:django_http_requests_total_by_view_transport_method:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_requests_total_by_view_transport_method[30s])) by (job,view,transport,method) | ||||||
|  |       - record: job:django_http_responses_total_by_templatename:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_total_by_templatename[30s])) by (job,templatename) | ||||||
|  |       - record: job:django_http_responses_total_by_status:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_total_by_status[30s])) by (job,status) | ||||||
|  |       - record: job:django_http_responses_total_by_status_name_method:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_total_by_status_name_method[30s])) by (job,status,name,method) | ||||||
|  |       - record: job:django_http_responses_total_by_charset:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_responses_total_by_charset[30s])) by (job,charset) | ||||||
|  |       - record: job:django_http_exceptions_total_by_type:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_exceptions_total_by_type[30s])) by (job,type) | ||||||
|  |       - record: job:django_http_exceptions_total_by_view:sum_rate30s | ||||||
|  |         expr: sum(rate(django_http_exceptions_total_by_view[30s])) by (job,view) | ||||||
|  |   - name: Aggregate latency histograms | ||||||
|  |     rules: | ||||||
|  |       - record: job:django_http_requests_latency_including_middlewares_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.50, sum(rate(django_http_requests_latency_including_middlewares_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "50" | ||||||
|  |       - record: job:django_http_requests_latency_including_middlewares_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.95, sum(rate(django_http_requests_latency_including_middlewares_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "95" | ||||||
|  |       - record: job:django_http_requests_latency_including_middlewares_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.99, sum(rate(django_http_requests_latency_including_middlewares_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "99" | ||||||
|  |       - record: job:django_http_requests_latency_including_middlewares_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.999, sum(rate(django_http_requests_latency_including_middlewares_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "99.9" | ||||||
|  |       - record: job:django_http_requests_latency_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.50, sum(rate(django_http_requests_latency_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "50" | ||||||
|  |       - record: job:django_http_requests_latency_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.95, sum(rate(django_http_requests_latency_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "95" | ||||||
|  |       - record: job:django_http_requests_latency_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.99, sum(rate(django_http_requests_latency_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "99" | ||||||
|  |       - record: job:django_http_requests_latency_seconds:quantile_rate30s | ||||||
|  |         expr: histogram_quantile(0.999, sum(rate(django_http_requests_latency_seconds_bucket[30s])) by (job, le)) | ||||||
|  |         labels: | ||||||
|  |           quantile: "99.9" | ||||||
|  |   - name: Aggregate model operations | ||||||
|  |     rules: | ||||||
|  |       - record: job:django_model_inserts_total:sum_rate1m | ||||||
|  |         expr: sum(rate(django_model_inserts_total[1m])) by (job, model) | ||||||
|  |       - record: job:django_model_updates_total:sum_rate1m | ||||||
|  |         expr: sum(rate(django_model_updates_total[1m])) by (job, model) | ||||||
|  |       - record: job:django_model_deletes_total:sum_rate1m | ||||||
|  |         expr: sum(rate(django_model_deletes_total[1m])) by (job, model) | ||||||
|  |   - name: Aggregate database operations | ||||||
|  |     rules: | ||||||
|  |       - record: job:django_db_new_connections_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_db_new_connections_total[30s])) by (alias, vendor) | ||||||
|  |       - record: job:django_db_new_connection_errors_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_db_new_connection_errors_total[30s])) by (alias, vendor) | ||||||
|  |       - record: job:django_db_execute_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_db_execute_total[30s])) by (alias, vendor) | ||||||
|  |       - record: job:django_db_execute_many_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_db_execute_many_total[30s])) by (alias, vendor) | ||||||
|  |       - record: job:django_db_errors_total:sum_rate30s | ||||||
|  |         expr: sum(rate(django_db_errors_total[30s])) by (alias, vendor, type) | ||||||
|  |   - name: Aggregate migrations | ||||||
|  |     rules: | ||||||
|  |       - record: job:django_migrations_applied_total:max | ||||||
|  |         expr: max(django_migrations_applied_total) by (job, connection) | ||||||
|  |       - record: job:django_migrations_unapplied_total:max | ||||||
|  |         expr: max(django_migrations_unapplied_total) by (job, connection) | ||||||
|  |   - name: Alerts | ||||||
|  |     rules: | ||||||
|  |       - alert: UnappliedMigrations | ||||||
|  |         expr: job:django_migrations_unapplied_total:max > 0 | ||||||
|  |         for: 1m | ||||||
|  |         labels: | ||||||
|  |           severity: testing | ||||||
|  |         annotations: | ||||||
|  |           summary: "Unapplied django migrations on {{$labels.connection}}" | ||||||
|  |           description: "Django detected {{$value}} unapplied migrations on database {{$labels.connection}}" | ||||||
|  | {{- end }} | ||||||
							
								
								
									
										12
									
								
								helm/passbook/templates/secret.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								helm/passbook/templates/secret.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | apiVersion: v1 | ||||||
|  | kind: Secret | ||||||
|  | type: Opaque | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  | data: | ||||||
|  |   monitoring_username: monitor | ||||||
|  |   {{- if .Values.config.secret_key }} | ||||||
|  |   secret_key: {{ .Values.config.secret_key | b64enc | quote }} | ||||||
|  |   {{- else }} | ||||||
|  |   secret_key: {{ randAlphaNum 50 | b64enc | quote}} | ||||||
|  |   {{- end }} | ||||||
							
								
								
									
										51
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | apiVersion: apps/v1 | ||||||
|  | kind: Deployment | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-static | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  | spec: | ||||||
|  |   selector: | ||||||
|  |     matchLabels: | ||||||
|  |       app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |       app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |   template: | ||||||
|  |     metadata: | ||||||
|  |       labels: | ||||||
|  |         app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |         app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |         k8s.passbook.io/component: static | ||||||
|  |     spec: | ||||||
|  |       containers: | ||||||
|  |         - name: {{ .Chart.Name }}-static | ||||||
|  |           image: "docker.beryju.org/passbook/static:{{ .Values.image.tag }}" | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
|  |           ports: | ||||||
|  |             - name: http | ||||||
|  |               containerPort: 80 | ||||||
|  |               protocol: TCP | ||||||
|  |           livenessProbe: | ||||||
|  |             initialDelaySeconds: 10 | ||||||
|  |             timeoutSeconds: 5 | ||||||
|  |             httpGet: | ||||||
|  |               path: /_/healthz | ||||||
|  |               port: http | ||||||
|  |           readinessProbe: | ||||||
|  |             initialDelaySeconds: 10 | ||||||
|  |             timeoutSeconds: 5 | ||||||
|  |             httpGet: | ||||||
|  |               path: /_/healthz | ||||||
|  |               port: http | ||||||
|  |           resources: | ||||||
|  |             requests: | ||||||
|  |               cpu: 10m | ||||||
|  |               memory: 10M | ||||||
|  |             limits: | ||||||
|  |               cpu: 20m | ||||||
|  |               memory: 20M | ||||||
|  |         - name: {{ .Chart.Name }}-static-prometheus | ||||||
|  |           image: nginx/nginx-prometheus-exporter:0.4.1 | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
							
								
								
									
										21
									
								
								helm/passbook/templates/static-service.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								helm/passbook/templates/static-service.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | apiVersion: v1 | ||||||
|  | kind: Service | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-static | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  |     k8s.passbook.io/component: static | ||||||
|  | spec: | ||||||
|  |   type: ClusterIP | ||||||
|  |   ports: | ||||||
|  |     - port: 80 | ||||||
|  |       targetPort: http | ||||||
|  |       protocol: TCP | ||||||
|  |       name: http | ||||||
|  |   selector: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     k8s.passbook.io/component: static | ||||||
							
								
								
									
										17
									
								
								helm/passbook/templates/static-sm.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								helm/passbook/templates/static-sm.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | {{- if .Values.monitoring.enabled -}} | ||||||
|  | apiVersion: monitoring.coreos.com/v1 | ||||||
|  | kind: ServiceMonitor | ||||||
|  | metadata: | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  |   name: {{ include "passbook.fullname" . }}-static-monitoring | ||||||
|  | spec: | ||||||
|  |   endpoints: | ||||||
|  |   - port: http | ||||||
|  |   selector: | ||||||
|  |     matchLabels: | ||||||
|  |       k8s.passbook.io/component: static | ||||||
|  | {{- end }} | ||||||
							
								
								
									
										112
									
								
								helm/passbook/templates/web-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								helm/passbook/templates/web-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,112 @@ | |||||||
|  | apiVersion: apps/v1 | ||||||
|  | kind: Deployment | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-web | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  | spec: | ||||||
|  |   replicas: 2 | ||||||
|  |   selector: | ||||||
|  |     matchLabels: | ||||||
|  |       app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |       app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |   template: | ||||||
|  |     metadata: | ||||||
|  |       labels: | ||||||
|  |         app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |         app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |         passbook.io/component: web | ||||||
|  |     spec: | ||||||
|  |       volumes: | ||||||
|  |         - name: config-volume | ||||||
|  |           configMap: | ||||||
|  |             name: {{ include "passbook.fullname" . }}-config | ||||||
|  |       initContainers: | ||||||
|  |         - name: passbook-database-migrations | ||||||
|  |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|  |           command: | ||||||
|  |             - ./manage.py | ||||||
|  |           args: | ||||||
|  |             - migrate | ||||||
|  |           volumeMounts: | ||||||
|  |             - mountPath: /etc/passbook | ||||||
|  |               name: config-volume | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - name: PASSBOOK_SECRET_KEY | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  |                   key: secret_key | ||||||
|  |             - name: PASSBOOK_REDIS__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-redis" | ||||||
|  |                   key: redis-password | ||||||
|  |             - name: PASSBOOK_POSTGRESQL__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-postgresql" | ||||||
|  |                   key: postgresql-password | ||||||
|  |       containers: | ||||||
|  |         - name: {{ .Chart.Name }} | ||||||
|  |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
|  |           command: | ||||||
|  |             - uwsgi | ||||||
|  |           args: | ||||||
|  |             - uwsgi.ini | ||||||
|  |           volumeMounts: | ||||||
|  |             - mountPath: /etc/passbook | ||||||
|  |               name: config-volume | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - name: PASSBOOK_SECRET_KEY | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  |                   key: secret_key | ||||||
|  |             - name: PASSBOOK_REDIS__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-redis" | ||||||
|  |                   key: redis-password | ||||||
|  |             - name: PASSBOOK_POSTGRESQL__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-postgresql" | ||||||
|  |                   key: postgresql-password | ||||||
|  |           ports: | ||||||
|  |             - name: http | ||||||
|  |               containerPort: 8000 | ||||||
|  |               protocol: TCP | ||||||
|  |           livenessProbe: | ||||||
|  |             httpGet: | ||||||
|  |               path: / | ||||||
|  |               port: http | ||||||
|  |               httpHeaders: | ||||||
|  |                 - name: Host | ||||||
|  |                   value: kubernetes-healthcheck-host | ||||||
|  |           readinessProbe: | ||||||
|  |             httpGet: | ||||||
|  |               path: / | ||||||
|  |               port: http | ||||||
|  |               httpHeaders: | ||||||
|  |                 - name: Host | ||||||
|  |                   value: kubernetes-healthcheck-host | ||||||
|  |           resources: | ||||||
|  |             requests: | ||||||
|  |               cpu: 100m | ||||||
|  |               memory: 200M | ||||||
|  |             limits: | ||||||
|  |               cpu: 300m | ||||||
|  |               memory: 350M | ||||||
| @ -1,19 +1,21 @@ | |||||||
| apiVersion: v1 | apiVersion: v1 | ||||||
| kind: Service | kind: Service | ||||||
| metadata: | metadata: | ||||||
|   name: {{ include "passbook.fullname" . }} |   name: {{ include "passbook.fullname" . }}-web | ||||||
|   labels: |   labels: | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |  | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|     app.kubernetes.io/managed-by: {{ .Release.Service }} |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     k8s.passbook.io/component: web | ||||||
| spec: | spec: | ||||||
|   type: {{ .Values.service.type }} |   type: ClusterIP | ||||||
|   ports: |   ports: | ||||||
|     - port: {{ .Values.service.port }} |     - port: 80 | ||||||
|       targetPort: http |       targetPort: http | ||||||
|       protocol: TCP |       protocol: TCP | ||||||
|       name: http |       name: http | ||||||
|   selector: |   selector: | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     passbook.io/component: web | ||||||
							
								
								
									
										25
									
								
								helm/passbook/templates/web-sm.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								helm/passbook/templates/web-sm.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | {{- if .Values.monitoring.enabled -}} | ||||||
|  | apiVersion: monitoring.coreos.com/v1 | ||||||
|  | kind: ServiceMonitor | ||||||
|  | metadata: | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  |   name: {{ include "passbook.fullname" . }}-web-monitoring | ||||||
|  | spec: | ||||||
|  |   endpoints: | ||||||
|  |   - basicAuth: | ||||||
|  |       password: | ||||||
|  |         name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  |         key: secret_key | ||||||
|  |       username: | ||||||
|  |         name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  |         key: monitoring_username | ||||||
|  |     port: http | ||||||
|  |     interval: 10s | ||||||
|  |   selector: | ||||||
|  |     matchLabels: | ||||||
|  |       k8s.passbook.io/component: web | ||||||
|  | {{- end }} | ||||||
							
								
								
									
										69
									
								
								helm/passbook/templates/worker-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								helm/passbook/templates/worker-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | |||||||
|  | apiVersion: apps/v1 | ||||||
|  | kind: Deployment | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-worker | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  | spec: | ||||||
|  |   replicas: 1 | ||||||
|  |   selector: | ||||||
|  |     matchLabels: | ||||||
|  |       app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |       app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |   template: | ||||||
|  |     metadata: | ||||||
|  |       labels: | ||||||
|  |         app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |         app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |         passbook.io/component: worker | ||||||
|  |     spec: | ||||||
|  |       volumes: | ||||||
|  |         - name: config-volume | ||||||
|  |           configMap: | ||||||
|  |             name: {{ include "passbook.fullname" . }}-config | ||||||
|  |       containers: | ||||||
|  |         - name: {{ .Chart.Name }} | ||||||
|  |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
|  |           command: | ||||||
|  |             - celery | ||||||
|  |           args: | ||||||
|  |             - worker | ||||||
|  |             - --autoscale=10,3 | ||||||
|  |             - -E | ||||||
|  |             - -B | ||||||
|  |             - -A=passbook.root.celery | ||||||
|  |             - -s=/tmp/celerybeat-schedule | ||||||
|  |           volumeMounts: | ||||||
|  |             - mountPath: /etc/passbook | ||||||
|  |               name: config-volume | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - name: PASSBOOK_SECRET_KEY | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: {{ include "passbook.fullname" . }}-secret-key | ||||||
|  |                   key: secret_key | ||||||
|  |             - name: PASSBOOK_REDIS__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-redis" | ||||||
|  |                   key: redis-password | ||||||
|  |             - name: PASSBOOK_POSTGRESQL__PASSWORD | ||||||
|  |               valueFrom: | ||||||
|  |                 secretKeyRef: | ||||||
|  |                   name: "{{ .Release.Name }}-postgresql" | ||||||
|  |                   key: postgresql-password | ||||||
|  |           resources: | ||||||
|  |             requests: | ||||||
|  |               cpu: 150m | ||||||
|  |               memory: 300M | ||||||
|  |             limits: | ||||||
|  |               cpu: 300m | ||||||
|  |               memory: 500M | ||||||
| @ -1,11 +1,8 @@ | |||||||
| # Default values for passbook. | # Default values for passbook. | ||||||
| # This is a YAML-formatted file. | # This is a YAML-formatted file. | ||||||
| # Declare variables to be passed into your templates. | # Declare variables to be passed into your templates. | ||||||
|  |  | ||||||
| replicaCount: 1 |  | ||||||
|  |  | ||||||
| image: | image: | ||||||
|   tag: latest |   tag: 0.7.2-beta | ||||||
|  |  | ||||||
| nameOverride: "" | nameOverride: "" | ||||||
|  |  | ||||||
| @ -14,14 +11,13 @@ config: | |||||||
|   # secret_key: _k*@6h2u2@q-dku57hhgzb7tnx*ba9wodcb^s9g0j59@=y(@_o |   # secret_key: _k*@6h2u2@q-dku57hhgzb7tnx*ba9wodcb^s9g0j59@=y(@_o | ||||||
|   # Enable error reporting |   # Enable error reporting | ||||||
|   error_reporting: true |   error_reporting: true | ||||||
|  |   email: | ||||||
|  |     host: localhost | ||||||
|  |  | ||||||
| postgresql: | # This Helm chart ships with built-in Prometheus ServiceMonitors and Rules. | ||||||
|     postgresqlDatabase: passbook | # This requires the CoreOS Prometheus Operator. | ||||||
|     postgresqlPassword: foo | monitoring: | ||||||
|  |   enabled: false | ||||||
| service: |  | ||||||
|   type: ClusterIP |  | ||||||
|   port: 80 |  | ||||||
|  |  | ||||||
| ingress: | ingress: | ||||||
|   enabled: false |   enabled: false | ||||||
| @ -31,27 +27,18 @@ ingress: | |||||||
|   path: / |   path: / | ||||||
|   hosts: |   hosts: | ||||||
|     - passbook.k8s.local |     - passbook.k8s.local | ||||||
|     - kubernetes-healthcheck-host |  | ||||||
|   defaultHost: passbook.k8s.local |  | ||||||
|   tls: [] |   tls: [] | ||||||
|   #  - secretName: chart-example-tls |   #  - secretName: chart-example-tls | ||||||
|   #    hosts: |   #    hosts: | ||||||
|   #      - passbook.k8s.local |   #      - passbook.k8s.local | ||||||
|  |  | ||||||
| resources: {} | # These settings configure the packaged PostgreSQL and Redis chart. | ||||||
|   # We usually recommend not to specify default resources and to leave this as a conscious | postgresql: | ||||||
|   # choice for the user. This also increases chances charts run on environments with little |   postgresqlDatabase: passbook | ||||||
|   # resources, such as Minikube. If you do want to specify resources, uncomment the following |  | ||||||
|   # lines, adjust them as necessary, and remove the curly braces after 'resources:'. |  | ||||||
|   # limits: |  | ||||||
|   #  cpu: 100m |  | ||||||
|   #  memory: 128Mi |  | ||||||
|   # requests: |  | ||||||
|   #  cpu: 100m |  | ||||||
|   #  memory: 128Mi |  | ||||||
|  |  | ||||||
| nodeSelector: {} | redis: | ||||||
|  |   cluster: | ||||||
| tolerations: [] |     enabled: false | ||||||
|  |   master: | ||||||
| affinity: {} |     persistence: | ||||||
|  |       enabled: false | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ import os | |||||||
| import sys | import sys | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.core.settings') |     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.root.settings') | ||||||
|     try: |     try: | ||||||
|         from django.core.management import execute_from_command_line |         from django.core.management import execute_from_command_line | ||||||
|     except ImportError as exc: |     except ImportError as exc: | ||||||
|  | |||||||
| @ -1,2 +1,2 @@ | |||||||
| """passbook""" | """passbook""" | ||||||
| __version__ = '0.0.8-alpha' | __version__ = '0.7.2-beta' | ||||||
|  | |||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook admin""" |  | ||||||
| __version__ = '0.0.8-alpha' |  | ||||||
|  | |||||||
| @ -1,6 +0,0 @@ | |||||||
| """Versioned Admin API Urls""" |  | ||||||
| from django.conf.urls import include, url |  | ||||||
|  |  | ||||||
| urlpatterns = [ |  | ||||||
|     url(r'^v1/', include('passbook.admin.api.v1.urls', namespace='v1')), |  | ||||||
| ] |  | ||||||
| @ -1,36 +0,0 @@ | |||||||
| """passbook admin gorup API""" |  | ||||||
| from rest_framework.permissions import IsAdminUser |  | ||||||
| from rest_framework.serializers import ModelSerializer, Serializer |  | ||||||
| from rest_framework.viewsets import ModelViewSet |  | ||||||
|  |  | ||||||
| from passbook.core.models import Group |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RecursiveField(Serializer): |  | ||||||
|     """Recursive field for manytomanyfield""" |  | ||||||
|  |  | ||||||
|     def to_representation(self, value): |  | ||||||
|         serializer = self.parent.parent.__class__(value, context=self.context) |  | ||||||
|         return serializer.data |  | ||||||
|  |  | ||||||
|     def create(self): |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
|     def update(self): |  | ||||||
|         raise NotImplementedError() |  | ||||||
|  |  | ||||||
| class GroupSerializer(ModelSerializer): |  | ||||||
|     """Group Serializer""" |  | ||||||
|  |  | ||||||
|     children = RecursiveField(many=True) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = Group |  | ||||||
|         fields = '__all__' |  | ||||||
|  |  | ||||||
| class GroupViewSet(ModelViewSet): |  | ||||||
|     """Group Viewset""" |  | ||||||
|  |  | ||||||
|     permission_classes = [IsAdminUser] |  | ||||||
|     serializer_class = GroupSerializer |  | ||||||
|     queryset = Group.objects.filter(parent__isnull=True) |  | ||||||
| @ -1,33 +0,0 @@ | |||||||
| """passbook admin API URLs""" |  | ||||||
| from django.urls import path |  | ||||||
| from drf_yasg import openapi |  | ||||||
| from drf_yasg.views import get_schema_view |  | ||||||
| from rest_framework import permissions |  | ||||||
| from rest_framework.routers import DefaultRouter |  | ||||||
|  |  | ||||||
| from passbook.admin.api.v1.applications import ApplicationViewSet |  | ||||||
| from passbook.admin.api.v1.groups import GroupViewSet |  | ||||||
| from passbook.admin.api.v1.users import UserViewSet |  | ||||||
|  |  | ||||||
| router = DefaultRouter() |  | ||||||
| router.register('applications', ApplicationViewSet) |  | ||||||
| router.register('groups', GroupViewSet) |  | ||||||
| router.register('users', UserViewSet) |  | ||||||
|  |  | ||||||
| SchemaView = get_schema_view( |  | ||||||
|     openapi.Info( |  | ||||||
|         title="passbook Administration API", |  | ||||||
|         default_version='v1', |  | ||||||
|         description="Internal passbook API for Administration Interface", |  | ||||||
|         contact=openapi.Contact(email="contact@snippets.local"), |  | ||||||
|         license=openapi.License(name="MIT License"), |  | ||||||
|     ), |  | ||||||
|     public=True, |  | ||||||
|     permission_classes=(permissions.IsAdminUser,), |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| urlpatterns = router.urls + [ |  | ||||||
|     path('swagger.yml', SchemaView.without_ui(cache_timeout=0), name='schema-json'), |  | ||||||
|     path('swagger/', SchemaView.with_ui('swagger', cache_timeout=0), name='schema-swagger-ui'), |  | ||||||
| ] |  | ||||||
| app_name = 'passbook.admin' |  | ||||||
							
								
								
									
										61
									
								
								passbook/admin/fields.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								passbook/admin/fields.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,61 @@ | |||||||
|  | """YAML fields""" | ||||||
|  | import yaml | ||||||
|  | from django import forms | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InvalidYAMLInput(str): | ||||||
|  |     """Invalid YAML String type""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class YAMLString(str): | ||||||
|  |     """YAML String type""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class YAMLField(forms.CharField): | ||||||
|  |     """Django's JSON Field converted to YAML""" | ||||||
|  |  | ||||||
|  |     default_error_messages = { | ||||||
|  |         'invalid': _("'%(value)s' value must be valid YAML."), | ||||||
|  |     } | ||||||
|  |     widget = forms.Textarea | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         if self.disabled: | ||||||
|  |             return value | ||||||
|  |         if value in self.empty_values: | ||||||
|  |             return None | ||||||
|  |         if isinstance(value, (list, dict, int, float, YAMLString)): | ||||||
|  |             return value | ||||||
|  |         try: | ||||||
|  |             converted = yaml.safe_load(value) | ||||||
|  |         except yaml.YAMLError: | ||||||
|  |             raise forms.ValidationError( | ||||||
|  |                 self.error_messages['invalid'], | ||||||
|  |                 code='invalid', | ||||||
|  |                 params={'value': value}, | ||||||
|  |             ) | ||||||
|  |         if isinstance(converted, str): | ||||||
|  |             return YAMLString(converted) | ||||||
|  |         return converted | ||||||
|  |  | ||||||
|  |     def bound_data(self, data, initial): | ||||||
|  |         if self.disabled: | ||||||
|  |             return initial | ||||||
|  |         try: | ||||||
|  |             return yaml.safe_load(data) | ||||||
|  |         except yaml.YAMLError: | ||||||
|  |             return InvalidYAMLInput(data) | ||||||
|  |  | ||||||
|  |     def prepare_value(self, value): | ||||||
|  |         if isinstance(value, InvalidYAMLInput): | ||||||
|  |             return value | ||||||
|  |         return yaml.dump(value, explicit_start=True) | ||||||
|  |  | ||||||
|  |     def has_changed(self, initial, data): | ||||||
|  |         if super().has_changed(initial, data): | ||||||
|  |             return True | ||||||
|  |         # For purposes of seeing whether something has changed, True isn't the | ||||||
|  |         # same as 1 and the order of keys doesn't matter. | ||||||
|  |         data = self.to_python(data) | ||||||
|  |         return yaml.dump(initial, sort_keys=True) != yaml.dump(data, sort_keys=True) | ||||||
							
								
								
									
										37
									
								
								passbook/admin/forms/base.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								passbook/admin/forms/base.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | |||||||
|  | """p2 form helpers""" | ||||||
|  | from django import forms | ||||||
|  |  | ||||||
|  | from passbook.admin.fields import YAMLField | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TagModelForm(forms.ModelForm): | ||||||
|  |     """Base form for models that have attributes""" | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         # Check if we have an instance, load tags otherwise use an empty dict | ||||||
|  |         instance = kwargs.get('instance', None) | ||||||
|  |         tags = instance.tags if instance else {} | ||||||
|  |         # Make sure all predefined tags exist in tags, and set default if they don't | ||||||
|  |         predefined_tags = self._meta.model().get_predefined_tags()  # pylint: disable=no-member | ||||||
|  |         for key, value in predefined_tags.items(): | ||||||
|  |             if key not in tags: | ||||||
|  |                 tags[key] = value | ||||||
|  |         # Format JSON | ||||||
|  |         kwargs['initial']['tags'] = tags | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def clean_tags(self): | ||||||
|  |         """Make sure all required tags are set""" | ||||||
|  |         if hasattr(self.instance, 'get_required_keys') and hasattr(self.instance, 'tags'): | ||||||
|  |             for key in self.instance.get_required_keys(): | ||||||
|  |                 if key not in self.cleaned_data.get('tags'): | ||||||
|  |                     raise forms.ValidationError("Tag %s missing." % key) | ||||||
|  |         return self.cleaned_data.get('tags') | ||||||
|  |  | ||||||
|  | # pylint: disable=too-few-public-methods | ||||||
|  | class TagModelFormMeta: | ||||||
|  |     """Base Meta class that uses the YAMLField""" | ||||||
|  |  | ||||||
|  |     field_classes = { | ||||||
|  |         'tags': YAMLField | ||||||
|  |     } | ||||||
| @ -2,5 +2,6 @@ | |||||||
| # from django import forms | # from django import forms | ||||||
|  |  | ||||||
| SOURCE_FORM_FIELDS = ['name', 'slug', 'enabled', 'policies'] | SOURCE_FORM_FIELDS = ['name', 'slug', 'enabled', 'policies'] | ||||||
|  | SOURCE_SERIALIZER_FIELDS = ['pk', 'name', 'slug', 'enabled', 'policies'] | ||||||
|  |  | ||||||
| # class SourceForm(forms.Form) | # class SourceForm(forms.Form) | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								passbook/admin/forms/users.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								passbook/admin/forms/users.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | """passbook administrative user forms""" | ||||||
|  |  | ||||||
|  | from django import forms | ||||||
|  |  | ||||||
|  | from passbook.admin.fields import YAMLField | ||||||
|  | from passbook.core.models import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UserForm(forms.ModelForm): | ||||||
|  |     """Update User Details""" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         model = User | ||||||
|  |         fields = ['username', 'name', 'email', 'is_staff', 'is_active', 'attributes'] | ||||||
|  |         widgets = { | ||||||
|  |             'name': forms.TextInput, | ||||||
|  |         } | ||||||
|  |         field_classes = { | ||||||
|  |             'attributes': YAMLField, | ||||||
|  |         } | ||||||
							
								
								
									
										25
									
								
								passbook/admin/middleware.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								passbook/admin/middleware.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | """passbook admin Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  | from passbook.core.models import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def impersonate(get_response): | ||||||
|  |     """Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  |     def middleware(request): | ||||||
|  |         """Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  |         # User is superuser and has __impersonate ID set | ||||||
|  |         if request.user.is_superuser and "__impersonate" in request.GET: | ||||||
|  |             request.session['impersonate_id'] = request.GET["__impersonate"] | ||||||
|  |         # user wants to stop impersonation | ||||||
|  |         elif "__unimpersonate" in request.GET and 'impersonate_id' in request.session: | ||||||
|  |             del request.session['impersonate_id'] | ||||||
|  |  | ||||||
|  |         # Actually impersonate user | ||||||
|  |         if request.user.is_superuser and 'impersonate_id' in request.session: | ||||||
|  |             request.user = User.objects.get(pk=request.session['impersonate_id']) | ||||||
|  |  | ||||||
|  |         response = get_response(request) | ||||||
|  |         return response | ||||||
|  |     return middleware | ||||||
| @ -1,2 +0,0 @@ | |||||||
| django-rest-framework |  | ||||||
| drf_yasg |  | ||||||
							
								
								
									
										5
									
								
								passbook/admin/settings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								passbook/admin/settings.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | """passbook admin settings""" | ||||||
|  |  | ||||||
|  | MIDDLEWARE = [ | ||||||
|  |     'passbook.admin.middleware.impersonate', | ||||||
|  | ] | ||||||
							
								
								
									
										209
									
								
								passbook/admin/static/codemirror/addon/comment/comment.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								passbook/admin/static/codemirror/addon/comment/comment.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,209 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var noOptions = {}; | ||||||
|  |   var nonWS = /[^\s\u00a0]/; | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   function firstNonWS(str) { | ||||||
|  |     var found = str.search(nonWS); | ||||||
|  |     return found == -1 ? 0 : found; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.toggleComment = function(cm) { | ||||||
|  |     cm.toggleComment(); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("toggleComment", function(options) { | ||||||
|  |     if (!options) options = noOptions; | ||||||
|  |     var cm = this; | ||||||
|  |     var minLine = Infinity, ranges = this.listSelections(), mode = null; | ||||||
|  |     for (var i = ranges.length - 1; i >= 0; i--) { | ||||||
|  |       var from = ranges[i].from(), to = ranges[i].to(); | ||||||
|  |       if (from.line >= minLine) continue; | ||||||
|  |       if (to.line >= minLine) to = Pos(minLine, 0); | ||||||
|  |       minLine = from.line; | ||||||
|  |       if (mode == null) { | ||||||
|  |         if (cm.uncomment(from, to, options)) mode = "un"; | ||||||
|  |         else { cm.lineComment(from, to, options); mode = "line"; } | ||||||
|  |       } else if (mode == "un") { | ||||||
|  |         cm.uncomment(from, to, options); | ||||||
|  |       } else { | ||||||
|  |         cm.lineComment(from, to, options); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Rough heuristic to try and detect lines that are part of multi-line string | ||||||
|  |   function probablyInsideString(cm, pos, line) { | ||||||
|  |     return /\bstring\b/.test(cm.getTokenTypeAt(Pos(pos.line, 0))) && !/^[\'\"\`]/.test(line) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getMode(cm, pos) { | ||||||
|  |     var mode = cm.getMode() | ||||||
|  |     return mode.useInnerComments === false || !mode.innerMode ? mode : cm.getModeAt(pos) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("lineComment", function(from, to, options) { | ||||||
|  |     if (!options) options = noOptions; | ||||||
|  |     var self = this, mode = getMode(self, from); | ||||||
|  |     var firstLine = self.getLine(from.line); | ||||||
|  |     if (firstLine == null || probablyInsideString(self, from, firstLine)) return; | ||||||
|  |  | ||||||
|  |     var commentString = options.lineComment || mode.lineComment; | ||||||
|  |     if (!commentString) { | ||||||
|  |       if (options.blockCommentStart || mode.blockCommentStart) { | ||||||
|  |         options.fullLines = true; | ||||||
|  |         self.blockComment(from, to, options); | ||||||
|  |       } | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var end = Math.min(to.ch != 0 || to.line == from.line ? to.line + 1 : to.line, self.lastLine() + 1); | ||||||
|  |     var pad = options.padding == null ? " " : options.padding; | ||||||
|  |     var blankLines = options.commentBlankLines || from.line == to.line; | ||||||
|  |  | ||||||
|  |     self.operation(function() { | ||||||
|  |       if (options.indent) { | ||||||
|  |         var baseString = null; | ||||||
|  |         for (var i = from.line; i < end; ++i) { | ||||||
|  |           var line = self.getLine(i); | ||||||
|  |           var whitespace = line.slice(0, firstNonWS(line)); | ||||||
|  |           if (baseString == null || baseString.length > whitespace.length) { | ||||||
|  |             baseString = whitespace; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         for (var i = from.line; i < end; ++i) { | ||||||
|  |           var line = self.getLine(i), cut = baseString.length; | ||||||
|  |           if (!blankLines && !nonWS.test(line)) continue; | ||||||
|  |           if (line.slice(0, cut) != baseString) cut = firstNonWS(line); | ||||||
|  |           self.replaceRange(baseString + commentString + pad, Pos(i, 0), Pos(i, cut)); | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         for (var i = from.line; i < end; ++i) { | ||||||
|  |           if (blankLines || nonWS.test(self.getLine(i))) | ||||||
|  |             self.replaceRange(commentString + pad, Pos(i, 0)); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("blockComment", function(from, to, options) { | ||||||
|  |     if (!options) options = noOptions; | ||||||
|  |     var self = this, mode = getMode(self, from); | ||||||
|  |     var startString = options.blockCommentStart || mode.blockCommentStart; | ||||||
|  |     var endString = options.blockCommentEnd || mode.blockCommentEnd; | ||||||
|  |     if (!startString || !endString) { | ||||||
|  |       if ((options.lineComment || mode.lineComment) && options.fullLines != false) | ||||||
|  |         self.lineComment(from, to, options); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     if (/\bcomment\b/.test(self.getTokenTypeAt(Pos(from.line, 0)))) return | ||||||
|  |  | ||||||
|  |     var end = Math.min(to.line, self.lastLine()); | ||||||
|  |     if (end != from.line && to.ch == 0 && nonWS.test(self.getLine(end))) --end; | ||||||
|  |  | ||||||
|  |     var pad = options.padding == null ? " " : options.padding; | ||||||
|  |     if (from.line > end) return; | ||||||
|  |  | ||||||
|  |     self.operation(function() { | ||||||
|  |       if (options.fullLines != false) { | ||||||
|  |         var lastLineHasText = nonWS.test(self.getLine(end)); | ||||||
|  |         self.replaceRange(pad + endString, Pos(end)); | ||||||
|  |         self.replaceRange(startString + pad, Pos(from.line, 0)); | ||||||
|  |         var lead = options.blockCommentLead || mode.blockCommentLead; | ||||||
|  |         if (lead != null) for (var i = from.line + 1; i <= end; ++i) | ||||||
|  |           if (i != end || lastLineHasText) | ||||||
|  |             self.replaceRange(lead + pad, Pos(i, 0)); | ||||||
|  |       } else { | ||||||
|  |         self.replaceRange(endString, to); | ||||||
|  |         self.replaceRange(startString, from); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("uncomment", function(from, to, options) { | ||||||
|  |     if (!options) options = noOptions; | ||||||
|  |     var self = this, mode = getMode(self, from); | ||||||
|  |     var end = Math.min(to.ch != 0 || to.line == from.line ? to.line : to.line - 1, self.lastLine()), start = Math.min(from.line, end); | ||||||
|  |  | ||||||
|  |     // Try finding line comments | ||||||
|  |     var lineString = options.lineComment || mode.lineComment, lines = []; | ||||||
|  |     var pad = options.padding == null ? " " : options.padding, didSomething; | ||||||
|  |     lineComment: { | ||||||
|  |       if (!lineString) break lineComment; | ||||||
|  |       for (var i = start; i <= end; ++i) { | ||||||
|  |         var line = self.getLine(i); | ||||||
|  |         var found = line.indexOf(lineString); | ||||||
|  |         if (found > -1 && !/comment/.test(self.getTokenTypeAt(Pos(i, found + 1)))) found = -1; | ||||||
|  |         if (found == -1 && nonWS.test(line)) break lineComment; | ||||||
|  |         if (found > -1 && nonWS.test(line.slice(0, found))) break lineComment; | ||||||
|  |         lines.push(line); | ||||||
|  |       } | ||||||
|  |       self.operation(function() { | ||||||
|  |         for (var i = start; i <= end; ++i) { | ||||||
|  |           var line = lines[i - start]; | ||||||
|  |           var pos = line.indexOf(lineString), endPos = pos + lineString.length; | ||||||
|  |           if (pos < 0) continue; | ||||||
|  |           if (line.slice(endPos, endPos + pad.length) == pad) endPos += pad.length; | ||||||
|  |           didSomething = true; | ||||||
|  |           self.replaceRange("", Pos(i, pos), Pos(i, endPos)); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |       if (didSomething) return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Try block comments | ||||||
|  |     var startString = options.blockCommentStart || mode.blockCommentStart; | ||||||
|  |     var endString = options.blockCommentEnd || mode.blockCommentEnd; | ||||||
|  |     if (!startString || !endString) return false; | ||||||
|  |     var lead = options.blockCommentLead || mode.blockCommentLead; | ||||||
|  |     var startLine = self.getLine(start), open = startLine.indexOf(startString) | ||||||
|  |     if (open == -1) return false | ||||||
|  |     var endLine = end == start ? startLine : self.getLine(end) | ||||||
|  |     var close = endLine.indexOf(endString, end == start ? open + startString.length : 0); | ||||||
|  |     var insideStart = Pos(start, open + 1), insideEnd = Pos(end, close + 1) | ||||||
|  |     if (close == -1 || | ||||||
|  |         !/comment/.test(self.getTokenTypeAt(insideStart)) || | ||||||
|  |         !/comment/.test(self.getTokenTypeAt(insideEnd)) || | ||||||
|  |         self.getRange(insideStart, insideEnd, "\n").indexOf(endString) > -1) | ||||||
|  |       return false; | ||||||
|  |  | ||||||
|  |     // Avoid killing block comments completely outside the selection. | ||||||
|  |     // Positions of the last startString before the start of the selection, and the first endString after it. | ||||||
|  |     var lastStart = startLine.lastIndexOf(startString, from.ch); | ||||||
|  |     var firstEnd = lastStart == -1 ? -1 : startLine.slice(0, from.ch).indexOf(endString, lastStart + startString.length); | ||||||
|  |     if (lastStart != -1 && firstEnd != -1 && firstEnd + endString.length != from.ch) return false; | ||||||
|  |     // Positions of the first endString after the end of the selection, and the last startString before it. | ||||||
|  |     firstEnd = endLine.indexOf(endString, to.ch); | ||||||
|  |     var almostLastStart = endLine.slice(to.ch).lastIndexOf(startString, firstEnd - to.ch); | ||||||
|  |     lastStart = (firstEnd == -1 || almostLastStart == -1) ? -1 : to.ch + almostLastStart; | ||||||
|  |     if (firstEnd != -1 && lastStart != -1 && lastStart != to.ch) return false; | ||||||
|  |  | ||||||
|  |     self.operation(function() { | ||||||
|  |       self.replaceRange("", Pos(end, close - (pad && endLine.slice(close - pad.length, close) == pad ? pad.length : 0)), | ||||||
|  |                         Pos(end, close + endString.length)); | ||||||
|  |       var openEnd = open + startString.length; | ||||||
|  |       if (pad && startLine.slice(openEnd, openEnd + pad.length) == pad) openEnd += pad.length; | ||||||
|  |       self.replaceRange("", Pos(start, open), Pos(start, openEnd)); | ||||||
|  |       if (lead) for (var i = start + 1; i <= end; ++i) { | ||||||
|  |         var line = self.getLine(i), found = line.indexOf(lead); | ||||||
|  |         if (found == -1 || nonWS.test(line.slice(0, found))) continue; | ||||||
|  |         var foundEnd = found + lead.length; | ||||||
|  |         if (pad && line.slice(foundEnd, foundEnd + pad.length) == pad) foundEnd += pad.length; | ||||||
|  |         self.replaceRange("", Pos(i, found), Pos(i, foundEnd)); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     return true; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										78
									
								
								passbook/admin/static/codemirror/addon/comment/continuecomment.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								passbook/admin/static/codemirror/addon/comment/continuecomment.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,78 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   function continueComment(cm) { | ||||||
|  |     if (cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |     var ranges = cm.listSelections(), mode, inserts = []; | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       var pos = ranges[i].head | ||||||
|  |       if (!/\bcomment\b/.test(cm.getTokenTypeAt(pos))) return CodeMirror.Pass; | ||||||
|  |       var modeHere = cm.getModeAt(pos) | ||||||
|  |       if (!mode) mode = modeHere; | ||||||
|  |       else if (mode != modeHere) return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |       var insert = null; | ||||||
|  |       if (mode.blockCommentStart && mode.blockCommentContinue) { | ||||||
|  |         var line = cm.getLine(pos.line).slice(0, pos.ch) | ||||||
|  |         var end = line.lastIndexOf(mode.blockCommentEnd), found | ||||||
|  |         if (end != -1 && end == pos.ch - mode.blockCommentEnd.length) { | ||||||
|  |           // Comment ended, don't continue it | ||||||
|  |         } else if ((found = line.lastIndexOf(mode.blockCommentStart)) > -1 && found > end) { | ||||||
|  |           insert = line.slice(0, found) | ||||||
|  |           if (/\S/.test(insert)) { | ||||||
|  |             insert = "" | ||||||
|  |             for (var j = 0; j < found; ++j) insert += " " | ||||||
|  |           } | ||||||
|  |         } else if ((found = line.indexOf(mode.blockCommentContinue)) > -1 && !/\S/.test(line.slice(0, found))) { | ||||||
|  |           insert = line.slice(0, found) | ||||||
|  |         } | ||||||
|  |         if (insert != null) insert += mode.blockCommentContinue | ||||||
|  |       } | ||||||
|  |       if (insert == null && mode.lineComment && continueLineCommentEnabled(cm)) { | ||||||
|  |         var line = cm.getLine(pos.line), found = line.indexOf(mode.lineComment); | ||||||
|  |         if (found > -1) { | ||||||
|  |           insert = line.slice(0, found); | ||||||
|  |           if (/\S/.test(insert)) insert = null; | ||||||
|  |           else insert += mode.lineComment + line.slice(found + mode.lineComment.length).match(/^\s*/)[0]; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       if (insert == null) return CodeMirror.Pass; | ||||||
|  |       inserts[i] = "\n" + insert; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cm.operation(function() { | ||||||
|  |       for (var i = ranges.length - 1; i >= 0; i--) | ||||||
|  |         cm.replaceRange(inserts[i], ranges[i].from(), ranges[i].to(), "+insert"); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function continueLineCommentEnabled(cm) { | ||||||
|  |     var opt = cm.getOption("continueComments"); | ||||||
|  |     if (opt && typeof opt == "object") | ||||||
|  |       return opt.continueLineComment !== false; | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("continueComments", null, function(cm, val, prev) { | ||||||
|  |     if (prev && prev != CodeMirror.Init) | ||||||
|  |       cm.removeKeyMap("continueComment"); | ||||||
|  |     if (val) { | ||||||
|  |       var key = "Enter"; | ||||||
|  |       if (typeof val == "string") | ||||||
|  |         key = val; | ||||||
|  |       else if (typeof val == "object" && val.key) | ||||||
|  |         key = val.key; | ||||||
|  |       var map = {name: "continueComment"}; | ||||||
|  |       map[key] = continueComment; | ||||||
|  |       cm.addKeyMap(map); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										32
									
								
								passbook/admin/static/codemirror/addon/dialog/dialog.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								passbook/admin/static/codemirror/addon/dialog/dialog.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | .CodeMirror-dialog { | ||||||
|  |   position: absolute; | ||||||
|  |   left: 0; right: 0; | ||||||
|  |   background: inherit; | ||||||
|  |   z-index: 15; | ||||||
|  |   padding: .1em .8em; | ||||||
|  |   overflow: hidden; | ||||||
|  |   color: inherit; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .CodeMirror-dialog-top { | ||||||
|  |   border-bottom: 1px solid #eee; | ||||||
|  |   top: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .CodeMirror-dialog-bottom { | ||||||
|  |   border-top: 1px solid #eee; | ||||||
|  |   bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .CodeMirror-dialog input { | ||||||
|  |   border: none; | ||||||
|  |   outline: none; | ||||||
|  |   background: transparent; | ||||||
|  |   width: 20em; | ||||||
|  |   color: inherit; | ||||||
|  |   font-family: monospace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .CodeMirror-dialog button { | ||||||
|  |   font-size: 70%; | ||||||
|  | } | ||||||
							
								
								
									
										161
									
								
								passbook/admin/static/codemirror/addon/dialog/dialog.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								passbook/admin/static/codemirror/addon/dialog/dialog.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,161 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | // Open simple dialogs on top of an editor. Relies on dialog.css. | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   function dialogDiv(cm, template, bottom) { | ||||||
|  |     var wrap = cm.getWrapperElement(); | ||||||
|  |     var dialog; | ||||||
|  |     dialog = wrap.appendChild(document.createElement("div")); | ||||||
|  |     if (bottom) | ||||||
|  |       dialog.className = "CodeMirror-dialog CodeMirror-dialog-bottom"; | ||||||
|  |     else | ||||||
|  |       dialog.className = "CodeMirror-dialog CodeMirror-dialog-top"; | ||||||
|  |  | ||||||
|  |     if (typeof template == "string") { | ||||||
|  |       dialog.innerHTML = template; | ||||||
|  |     } else { // Assuming it's a detached DOM element. | ||||||
|  |       dialog.appendChild(template); | ||||||
|  |     } | ||||||
|  |     CodeMirror.addClass(wrap, 'dialog-opened'); | ||||||
|  |     return dialog; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function closeNotification(cm, newVal) { | ||||||
|  |     if (cm.state.currentNotificationClose) | ||||||
|  |       cm.state.currentNotificationClose(); | ||||||
|  |     cm.state.currentNotificationClose = newVal; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("openDialog", function(template, callback, options) { | ||||||
|  |     if (!options) options = {}; | ||||||
|  |  | ||||||
|  |     closeNotification(this, null); | ||||||
|  |  | ||||||
|  |     var dialog = dialogDiv(this, template, options.bottom); | ||||||
|  |     var closed = false, me = this; | ||||||
|  |     function close(newVal) { | ||||||
|  |       if (typeof newVal == 'string') { | ||||||
|  |         inp.value = newVal; | ||||||
|  |       } else { | ||||||
|  |         if (closed) return; | ||||||
|  |         closed = true; | ||||||
|  |         CodeMirror.rmClass(dialog.parentNode, 'dialog-opened'); | ||||||
|  |         dialog.parentNode.removeChild(dialog); | ||||||
|  |         me.focus(); | ||||||
|  |  | ||||||
|  |         if (options.onClose) options.onClose(dialog); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var inp = dialog.getElementsByTagName("input")[0], button; | ||||||
|  |     if (inp) { | ||||||
|  |       inp.focus(); | ||||||
|  |  | ||||||
|  |       if (options.value) { | ||||||
|  |         inp.value = options.value; | ||||||
|  |         if (options.selectValueOnOpen !== false) { | ||||||
|  |           inp.select(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (options.onInput) | ||||||
|  |         CodeMirror.on(inp, "input", function(e) { options.onInput(e, inp.value, close);}); | ||||||
|  |       if (options.onKeyUp) | ||||||
|  |         CodeMirror.on(inp, "keyup", function(e) {options.onKeyUp(e, inp.value, close);}); | ||||||
|  |  | ||||||
|  |       CodeMirror.on(inp, "keydown", function(e) { | ||||||
|  |         if (options && options.onKeyDown && options.onKeyDown(e, inp.value, close)) { return; } | ||||||
|  |         if (e.keyCode == 27 || (options.closeOnEnter !== false && e.keyCode == 13)) { | ||||||
|  |           inp.blur(); | ||||||
|  |           CodeMirror.e_stop(e); | ||||||
|  |           close(); | ||||||
|  |         } | ||||||
|  |         if (e.keyCode == 13) callback(inp.value, e); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       if (options.closeOnBlur !== false) CodeMirror.on(inp, "blur", close); | ||||||
|  |     } else if (button = dialog.getElementsByTagName("button")[0]) { | ||||||
|  |       CodeMirror.on(button, "click", function() { | ||||||
|  |         close(); | ||||||
|  |         me.focus(); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       if (options.closeOnBlur !== false) CodeMirror.on(button, "blur", close); | ||||||
|  |  | ||||||
|  |       button.focus(); | ||||||
|  |     } | ||||||
|  |     return close; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("openConfirm", function(template, callbacks, options) { | ||||||
|  |     closeNotification(this, null); | ||||||
|  |     var dialog = dialogDiv(this, template, options && options.bottom); | ||||||
|  |     var buttons = dialog.getElementsByTagName("button"); | ||||||
|  |     var closed = false, me = this, blurring = 1; | ||||||
|  |     function close() { | ||||||
|  |       if (closed) return; | ||||||
|  |       closed = true; | ||||||
|  |       CodeMirror.rmClass(dialog.parentNode, 'dialog-opened'); | ||||||
|  |       dialog.parentNode.removeChild(dialog); | ||||||
|  |       me.focus(); | ||||||
|  |     } | ||||||
|  |     buttons[0].focus(); | ||||||
|  |     for (var i = 0; i < buttons.length; ++i) { | ||||||
|  |       var b = buttons[i]; | ||||||
|  |       (function(callback) { | ||||||
|  |         CodeMirror.on(b, "click", function(e) { | ||||||
|  |           CodeMirror.e_preventDefault(e); | ||||||
|  |           close(); | ||||||
|  |           if (callback) callback(me); | ||||||
|  |         }); | ||||||
|  |       })(callbacks[i]); | ||||||
|  |       CodeMirror.on(b, "blur", function() { | ||||||
|  |         --blurring; | ||||||
|  |         setTimeout(function() { if (blurring <= 0) close(); }, 200); | ||||||
|  |       }); | ||||||
|  |       CodeMirror.on(b, "focus", function() { ++blurring; }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   /* | ||||||
|  |    * openNotification | ||||||
|  |    * Opens a notification, that can be closed with an optional timer | ||||||
|  |    * (default 5000ms timer) and always closes on click. | ||||||
|  |    * | ||||||
|  |    * If a notification is opened while another is opened, it will close the | ||||||
|  |    * currently opened one and open the new one immediately. | ||||||
|  |    */ | ||||||
|  |   CodeMirror.defineExtension("openNotification", function(template, options) { | ||||||
|  |     closeNotification(this, close); | ||||||
|  |     var dialog = dialogDiv(this, template, options && options.bottom); | ||||||
|  |     var closed = false, doneTimer; | ||||||
|  |     var duration = options && typeof options.duration !== "undefined" ? options.duration : 5000; | ||||||
|  |  | ||||||
|  |     function close() { | ||||||
|  |       if (closed) return; | ||||||
|  |       closed = true; | ||||||
|  |       clearTimeout(doneTimer); | ||||||
|  |       CodeMirror.rmClass(dialog.parentNode, 'dialog-opened'); | ||||||
|  |       dialog.parentNode.removeChild(dialog); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     CodeMirror.on(dialog, 'click', function(e) { | ||||||
|  |       CodeMirror.e_preventDefault(e); | ||||||
|  |       close(); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     if (duration) | ||||||
|  |       doneTimer = setTimeout(close, duration); | ||||||
|  |  | ||||||
|  |     return close; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										47
									
								
								passbook/admin/static/codemirror/addon/display/autorefresh.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								passbook/admin/static/codemirror/addon/display/autorefresh.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,47 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")) | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod) | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror) | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict" | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("autoRefresh", false, function(cm, val) { | ||||||
|  |     if (cm.state.autoRefresh) { | ||||||
|  |       stopListening(cm, cm.state.autoRefresh) | ||||||
|  |       cm.state.autoRefresh = null | ||||||
|  |     } | ||||||
|  |     if (val && cm.display.wrapper.offsetHeight == 0) | ||||||
|  |       startListening(cm, cm.state.autoRefresh = {delay: val.delay || 250}) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   function startListening(cm, state) { | ||||||
|  |     function check() { | ||||||
|  |       if (cm.display.wrapper.offsetHeight) { | ||||||
|  |         stopListening(cm, state) | ||||||
|  |         if (cm.display.lastWrapHeight != cm.display.wrapper.clientHeight) | ||||||
|  |           cm.refresh() | ||||||
|  |       } else { | ||||||
|  |         state.timeout = setTimeout(check, state.delay) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     state.timeout = setTimeout(check, state.delay) | ||||||
|  |     state.hurry = function() { | ||||||
|  |       clearTimeout(state.timeout) | ||||||
|  |       state.timeout = setTimeout(check, 50) | ||||||
|  |     } | ||||||
|  |     CodeMirror.on(window, "mouseup", state.hurry) | ||||||
|  |     CodeMirror.on(window, "keyup", state.hurry) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function stopListening(_cm, state) { | ||||||
|  |     clearTimeout(state.timeout) | ||||||
|  |     CodeMirror.off(window, "mouseup", state.hurry) | ||||||
|  |     CodeMirror.off(window, "keyup", state.hurry) | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										6
									
								
								passbook/admin/static/codemirror/addon/display/fullscreen.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								passbook/admin/static/codemirror/addon/display/fullscreen.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | .CodeMirror-fullscreen { | ||||||
|  |   position: fixed; | ||||||
|  |   top: 0; left: 0; right: 0; bottom: 0; | ||||||
|  |   height: auto; | ||||||
|  |   z-index: 9; | ||||||
|  | } | ||||||
							
								
								
									
										41
									
								
								passbook/admin/static/codemirror/addon/display/fullscreen.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								passbook/admin/static/codemirror/addon/display/fullscreen.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("fullScreen", false, function(cm, val, old) { | ||||||
|  |     if (old == CodeMirror.Init) old = false; | ||||||
|  |     if (!old == !val) return; | ||||||
|  |     if (val) setFullscreen(cm); | ||||||
|  |     else setNormal(cm); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function setFullscreen(cm) { | ||||||
|  |     var wrap = cm.getWrapperElement(); | ||||||
|  |     cm.state.fullScreenRestore = {scrollTop: window.pageYOffset, scrollLeft: window.pageXOffset, | ||||||
|  |                                   width: wrap.style.width, height: wrap.style.height}; | ||||||
|  |     wrap.style.width = ""; | ||||||
|  |     wrap.style.height = "auto"; | ||||||
|  |     wrap.className += " CodeMirror-fullscreen"; | ||||||
|  |     document.documentElement.style.overflow = "hidden"; | ||||||
|  |     cm.refresh(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function setNormal(cm) { | ||||||
|  |     var wrap = cm.getWrapperElement(); | ||||||
|  |     wrap.className = wrap.className.replace(/\s*CodeMirror-fullscreen\b/, ""); | ||||||
|  |     document.documentElement.style.overflow = ""; | ||||||
|  |     var info = cm.state.fullScreenRestore; | ||||||
|  |     wrap.style.width = info.width; wrap.style.height = info.height; | ||||||
|  |     window.scrollTo(info.scrollLeft, info.scrollTop); | ||||||
|  |     cm.refresh(); | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										127
									
								
								passbook/admin/static/codemirror/addon/display/panel.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								passbook/admin/static/codemirror/addon/display/panel.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,127 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   CodeMirror.defineExtension("addPanel", function(node, options) { | ||||||
|  |     options = options || {}; | ||||||
|  |  | ||||||
|  |     if (!this.state.panels) initPanels(this); | ||||||
|  |  | ||||||
|  |     var info = this.state.panels; | ||||||
|  |     var wrapper = info.wrapper; | ||||||
|  |     var cmWrapper = this.getWrapperElement(); | ||||||
|  |     var replace = options.replace instanceof Panel && !options.replace.cleared; | ||||||
|  |  | ||||||
|  |     if (options.after instanceof Panel && !options.after.cleared) { | ||||||
|  |       wrapper.insertBefore(node, options.before.node.nextSibling); | ||||||
|  |     } else if (options.before instanceof Panel && !options.before.cleared) { | ||||||
|  |       wrapper.insertBefore(node, options.before.node); | ||||||
|  |     } else if (replace) { | ||||||
|  |       wrapper.insertBefore(node, options.replace.node); | ||||||
|  |       info.panels++; | ||||||
|  |       options.replace.clear(); | ||||||
|  |     } else if (options.position == "bottom") { | ||||||
|  |       wrapper.appendChild(node); | ||||||
|  |     } else if (options.position == "before-bottom") { | ||||||
|  |       wrapper.insertBefore(node, cmWrapper.nextSibling); | ||||||
|  |     } else if (options.position == "after-top") { | ||||||
|  |       wrapper.insertBefore(node, cmWrapper); | ||||||
|  |     } else { | ||||||
|  |       wrapper.insertBefore(node, wrapper.firstChild); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var height = (options && options.height) || node.offsetHeight; | ||||||
|  |     this._setSize(null, info.heightLeft -= height); | ||||||
|  |     if (!replace) { | ||||||
|  |       info.panels++; | ||||||
|  |     } | ||||||
|  |     if (options.stable && isAtTop(this, node)) | ||||||
|  |       this.scrollTo(null, this.getScrollInfo().top + height) | ||||||
|  |  | ||||||
|  |     return new Panel(this, node, options, height); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function Panel(cm, node, options, height) { | ||||||
|  |     this.cm = cm; | ||||||
|  |     this.node = node; | ||||||
|  |     this.options = options; | ||||||
|  |     this.height = height; | ||||||
|  |     this.cleared = false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Panel.prototype.clear = function() { | ||||||
|  |     if (this.cleared) return; | ||||||
|  |     this.cleared = true; | ||||||
|  |     var info = this.cm.state.panels; | ||||||
|  |     this.cm._setSize(null, info.heightLeft += this.height); | ||||||
|  |     if (this.options.stable && isAtTop(this.cm, this.node)) | ||||||
|  |       this.cm.scrollTo(null, this.cm.getScrollInfo().top - this.height) | ||||||
|  |     info.wrapper.removeChild(this.node); | ||||||
|  |     if (--info.panels == 0) removePanels(this.cm); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   Panel.prototype.changed = function(height) { | ||||||
|  |     var newHeight = height == null ? this.node.offsetHeight : height; | ||||||
|  |     var info = this.cm.state.panels; | ||||||
|  |     this.cm._setSize(null, info.heightLeft -= (newHeight - this.height)); | ||||||
|  |     this.height = newHeight; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   function initPanels(cm) { | ||||||
|  |     var wrap = cm.getWrapperElement(); | ||||||
|  |     var style = window.getComputedStyle ? window.getComputedStyle(wrap) : wrap.currentStyle; | ||||||
|  |     var height = parseInt(style.height); | ||||||
|  |     var info = cm.state.panels = { | ||||||
|  |       setHeight: wrap.style.height, | ||||||
|  |       heightLeft: height, | ||||||
|  |       panels: 0, | ||||||
|  |       wrapper: document.createElement("div") | ||||||
|  |     }; | ||||||
|  |     wrap.parentNode.insertBefore(info.wrapper, wrap); | ||||||
|  |     var hasFocus = cm.hasFocus(); | ||||||
|  |     info.wrapper.appendChild(wrap); | ||||||
|  |     if (hasFocus) cm.focus(); | ||||||
|  |  | ||||||
|  |     cm._setSize = cm.setSize; | ||||||
|  |     if (height != null) cm.setSize = function(width, newHeight) { | ||||||
|  |       if (newHeight == null) return this._setSize(width, newHeight); | ||||||
|  |       info.setHeight = newHeight; | ||||||
|  |       if (typeof newHeight != "number") { | ||||||
|  |         var px = /^(\d+\.?\d*)px$/.exec(newHeight); | ||||||
|  |         if (px) { | ||||||
|  |           newHeight = Number(px[1]); | ||||||
|  |         } else { | ||||||
|  |           info.wrapper.style.height = newHeight; | ||||||
|  |           newHeight = info.wrapper.offsetHeight; | ||||||
|  |           info.wrapper.style.height = ""; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       cm._setSize(width, info.heightLeft += (newHeight - height)); | ||||||
|  |       height = newHeight; | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function removePanels(cm) { | ||||||
|  |     var info = cm.state.panels; | ||||||
|  |     cm.state.panels = null; | ||||||
|  |  | ||||||
|  |     var wrap = cm.getWrapperElement(); | ||||||
|  |     info.wrapper.parentNode.replaceChild(wrap, info.wrapper); | ||||||
|  |     wrap.style.height = info.setHeight; | ||||||
|  |     cm.setSize = cm._setSize; | ||||||
|  |     cm.setSize(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function isAtTop(cm, dom) { | ||||||
|  |     for (var sibling = dom.nextSibling; sibling; sibling = sibling.nextSibling) | ||||||
|  |       if (sibling == cm.getWrapperElement()) return true | ||||||
|  |     return false | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										63
									
								
								passbook/admin/static/codemirror/addon/display/placeholder.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								passbook/admin/static/codemirror/addon/display/placeholder.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   CodeMirror.defineOption("placeholder", "", function(cm, val, old) { | ||||||
|  |     var prev = old && old != CodeMirror.Init; | ||||||
|  |     if (val && !prev) { | ||||||
|  |       cm.on("blur", onBlur); | ||||||
|  |       cm.on("change", onChange); | ||||||
|  |       cm.on("swapDoc", onChange); | ||||||
|  |       onChange(cm); | ||||||
|  |     } else if (!val && prev) { | ||||||
|  |       cm.off("blur", onBlur); | ||||||
|  |       cm.off("change", onChange); | ||||||
|  |       cm.off("swapDoc", onChange); | ||||||
|  |       clearPlaceholder(cm); | ||||||
|  |       var wrapper = cm.getWrapperElement(); | ||||||
|  |       wrapper.className = wrapper.className.replace(" CodeMirror-empty", ""); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (val && !cm.hasFocus()) onBlur(cm); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function clearPlaceholder(cm) { | ||||||
|  |     if (cm.state.placeholder) { | ||||||
|  |       cm.state.placeholder.parentNode.removeChild(cm.state.placeholder); | ||||||
|  |       cm.state.placeholder = null; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function setPlaceholder(cm) { | ||||||
|  |     clearPlaceholder(cm); | ||||||
|  |     var elt = cm.state.placeholder = document.createElement("pre"); | ||||||
|  |     elt.style.cssText = "height: 0; overflow: visible"; | ||||||
|  |     elt.style.direction = cm.getOption("direction"); | ||||||
|  |     elt.className = "CodeMirror-placeholder CodeMirror-line-like"; | ||||||
|  |     var placeHolder = cm.getOption("placeholder") | ||||||
|  |     if (typeof placeHolder == "string") placeHolder = document.createTextNode(placeHolder) | ||||||
|  |     elt.appendChild(placeHolder) | ||||||
|  |     cm.display.lineSpace.insertBefore(elt, cm.display.lineSpace.firstChild); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function onBlur(cm) { | ||||||
|  |     if (isEmpty(cm)) setPlaceholder(cm); | ||||||
|  |   } | ||||||
|  |   function onChange(cm) { | ||||||
|  |     var wrapper = cm.getWrapperElement(), empty = isEmpty(cm); | ||||||
|  |     wrapper.className = wrapper.className.replace(" CodeMirror-empty", "") + (empty ? " CodeMirror-empty" : ""); | ||||||
|  |  | ||||||
|  |     if (empty) setPlaceholder(cm); | ||||||
|  |     else clearPlaceholder(cm); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function isEmpty(cm) { | ||||||
|  |     return (cm.lineCount() === 1) && (cm.getLine(0) === ""); | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										51
									
								
								passbook/admin/static/codemirror/addon/display/rulers.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								passbook/admin/static/codemirror/addon/display/rulers.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("rulers", false, function(cm, val) { | ||||||
|  |     if (cm.state.rulerDiv) { | ||||||
|  |       cm.state.rulerDiv.parentElement.removeChild(cm.state.rulerDiv) | ||||||
|  |       cm.state.rulerDiv = null | ||||||
|  |       cm.off("refresh", drawRulers) | ||||||
|  |     } | ||||||
|  |     if (val && val.length) { | ||||||
|  |       cm.state.rulerDiv = cm.display.lineSpace.parentElement.insertBefore(document.createElement("div"), cm.display.lineSpace) | ||||||
|  |       cm.state.rulerDiv.className = "CodeMirror-rulers" | ||||||
|  |       drawRulers(cm) | ||||||
|  |       cm.on("refresh", drawRulers) | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function drawRulers(cm) { | ||||||
|  |     cm.state.rulerDiv.textContent = "" | ||||||
|  |     var val = cm.getOption("rulers"); | ||||||
|  |     var cw = cm.defaultCharWidth(); | ||||||
|  |     var left = cm.charCoords(CodeMirror.Pos(cm.firstLine(), 0), "div").left; | ||||||
|  |     cm.state.rulerDiv.style.minHeight = (cm.display.scroller.offsetHeight + 30) + "px"; | ||||||
|  |     for (var i = 0; i < val.length; i++) { | ||||||
|  |       var elt = document.createElement("div"); | ||||||
|  |       elt.className = "CodeMirror-ruler"; | ||||||
|  |       var col, conf = val[i]; | ||||||
|  |       if (typeof conf == "number") { | ||||||
|  |         col = conf; | ||||||
|  |       } else { | ||||||
|  |         col = conf.column; | ||||||
|  |         if (conf.className) elt.className += " " + conf.className; | ||||||
|  |         if (conf.color) elt.style.borderColor = conf.color; | ||||||
|  |         if (conf.lineStyle) elt.style.borderLeftStyle = conf.lineStyle; | ||||||
|  |         if (conf.width) elt.style.borderLeftWidth = conf.width; | ||||||
|  |       } | ||||||
|  |       elt.style.left = (left + col * cw) + "px"; | ||||||
|  |       cm.state.rulerDiv.appendChild(elt) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										191
									
								
								passbook/admin/static/codemirror/addon/edit/closebrackets.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								passbook/admin/static/codemirror/addon/edit/closebrackets.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,191 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   var defaults = { | ||||||
|  |     pairs: "()[]{}''\"\"", | ||||||
|  |     closeBefore: ")]}'\":;>", | ||||||
|  |     triples: "", | ||||||
|  |     explode: "[]{}" | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("autoCloseBrackets", false, function(cm, val, old) { | ||||||
|  |     if (old && old != CodeMirror.Init) { | ||||||
|  |       cm.removeKeyMap(keyMap); | ||||||
|  |       cm.state.closeBrackets = null; | ||||||
|  |     } | ||||||
|  |     if (val) { | ||||||
|  |       ensureBound(getOption(val, "pairs")) | ||||||
|  |       cm.state.closeBrackets = val; | ||||||
|  |       cm.addKeyMap(keyMap); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function getOption(conf, name) { | ||||||
|  |     if (name == "pairs" && typeof conf == "string") return conf; | ||||||
|  |     if (typeof conf == "object" && conf[name] != null) return conf[name]; | ||||||
|  |     return defaults[name]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var keyMap = {Backspace: handleBackspace, Enter: handleEnter}; | ||||||
|  |   function ensureBound(chars) { | ||||||
|  |     for (var i = 0; i < chars.length; i++) { | ||||||
|  |       var ch = chars.charAt(i), key = "'" + ch + "'" | ||||||
|  |       if (!keyMap[key]) keyMap[key] = handler(ch) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   ensureBound(defaults.pairs + "`") | ||||||
|  |  | ||||||
|  |   function handler(ch) { | ||||||
|  |     return function(cm) { return handleChar(cm, ch); }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getConfig(cm) { | ||||||
|  |     var deflt = cm.state.closeBrackets; | ||||||
|  |     if (!deflt || deflt.override) return deflt; | ||||||
|  |     var mode = cm.getModeAt(cm.getCursor()); | ||||||
|  |     return mode.closeBrackets || deflt; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handleBackspace(cm) { | ||||||
|  |     var conf = getConfig(cm); | ||||||
|  |     if (!conf || cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |     var pairs = getOption(conf, "pairs"); | ||||||
|  |     var ranges = cm.listSelections(); | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       if (!ranges[i].empty()) return CodeMirror.Pass; | ||||||
|  |       var around = charsAround(cm, ranges[i].head); | ||||||
|  |       if (!around || pairs.indexOf(around) % 2 != 0) return CodeMirror.Pass; | ||||||
|  |     } | ||||||
|  |     for (var i = ranges.length - 1; i >= 0; i--) { | ||||||
|  |       var cur = ranges[i].head; | ||||||
|  |       cm.replaceRange("", Pos(cur.line, cur.ch - 1), Pos(cur.line, cur.ch + 1), "+delete"); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handleEnter(cm) { | ||||||
|  |     var conf = getConfig(cm); | ||||||
|  |     var explode = conf && getOption(conf, "explode"); | ||||||
|  |     if (!explode || cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |     var ranges = cm.listSelections(); | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       if (!ranges[i].empty()) return CodeMirror.Pass; | ||||||
|  |       var around = charsAround(cm, ranges[i].head); | ||||||
|  |       if (!around || explode.indexOf(around) % 2 != 0) return CodeMirror.Pass; | ||||||
|  |     } | ||||||
|  |     cm.operation(function() { | ||||||
|  |       var linesep = cm.lineSeparator() || "\n"; | ||||||
|  |       cm.replaceSelection(linesep + linesep, null); | ||||||
|  |       cm.execCommand("goCharLeft"); | ||||||
|  |       ranges = cm.listSelections(); | ||||||
|  |       for (var i = 0; i < ranges.length; i++) { | ||||||
|  |         var line = ranges[i].head.line; | ||||||
|  |         cm.indentLine(line, null, true); | ||||||
|  |         cm.indentLine(line + 1, null, true); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function contractSelection(sel) { | ||||||
|  |     var inverted = CodeMirror.cmpPos(sel.anchor, sel.head) > 0; | ||||||
|  |     return {anchor: new Pos(sel.anchor.line, sel.anchor.ch + (inverted ? -1 : 1)), | ||||||
|  |             head: new Pos(sel.head.line, sel.head.ch + (inverted ? 1 : -1))}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handleChar(cm, ch) { | ||||||
|  |     var conf = getConfig(cm); | ||||||
|  |     if (!conf || cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |     var pairs = getOption(conf, "pairs"); | ||||||
|  |     var pos = pairs.indexOf(ch); | ||||||
|  |     if (pos == -1) return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |     var closeBefore = getOption(conf,"closeBefore"); | ||||||
|  |  | ||||||
|  |     var triples = getOption(conf, "triples"); | ||||||
|  |  | ||||||
|  |     var identical = pairs.charAt(pos + 1) == ch; | ||||||
|  |     var ranges = cm.listSelections(); | ||||||
|  |     var opening = pos % 2 == 0; | ||||||
|  |  | ||||||
|  |     var type; | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       var range = ranges[i], cur = range.head, curType; | ||||||
|  |       var next = cm.getRange(cur, Pos(cur.line, cur.ch + 1)); | ||||||
|  |       if (opening && !range.empty()) { | ||||||
|  |         curType = "surround"; | ||||||
|  |       } else if ((identical || !opening) && next == ch) { | ||||||
|  |         if (identical && stringStartsAfter(cm, cur)) | ||||||
|  |           curType = "both"; | ||||||
|  |         else if (triples.indexOf(ch) >= 0 && cm.getRange(cur, Pos(cur.line, cur.ch + 3)) == ch + ch + ch) | ||||||
|  |           curType = "skipThree"; | ||||||
|  |         else | ||||||
|  |           curType = "skip"; | ||||||
|  |       } else if (identical && cur.ch > 1 && triples.indexOf(ch) >= 0 && | ||||||
|  |                  cm.getRange(Pos(cur.line, cur.ch - 2), cur) == ch + ch) { | ||||||
|  |         if (cur.ch > 2 && /\bstring/.test(cm.getTokenTypeAt(Pos(cur.line, cur.ch - 2)))) return CodeMirror.Pass; | ||||||
|  |         curType = "addFour"; | ||||||
|  |       } else if (identical) { | ||||||
|  |         var prev = cur.ch == 0 ? " " : cm.getRange(Pos(cur.line, cur.ch - 1), cur) | ||||||
|  |         if (!CodeMirror.isWordChar(next) && prev != ch && !CodeMirror.isWordChar(prev)) curType = "both"; | ||||||
|  |         else return CodeMirror.Pass; | ||||||
|  |       } else if (opening && (next.length === 0 || /\s/.test(next) || closeBefore.indexOf(next) > -1)) { | ||||||
|  |         curType = "both"; | ||||||
|  |       } else { | ||||||
|  |         return CodeMirror.Pass; | ||||||
|  |       } | ||||||
|  |       if (!type) type = curType; | ||||||
|  |       else if (type != curType) return CodeMirror.Pass; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var left = pos % 2 ? pairs.charAt(pos - 1) : ch; | ||||||
|  |     var right = pos % 2 ? ch : pairs.charAt(pos + 1); | ||||||
|  |     cm.operation(function() { | ||||||
|  |       if (type == "skip") { | ||||||
|  |         cm.execCommand("goCharRight"); | ||||||
|  |       } else if (type == "skipThree") { | ||||||
|  |         for (var i = 0; i < 3; i++) | ||||||
|  |           cm.execCommand("goCharRight"); | ||||||
|  |       } else if (type == "surround") { | ||||||
|  |         var sels = cm.getSelections(); | ||||||
|  |         for (var i = 0; i < sels.length; i++) | ||||||
|  |           sels[i] = left + sels[i] + right; | ||||||
|  |         cm.replaceSelections(sels, "around"); | ||||||
|  |         sels = cm.listSelections().slice(); | ||||||
|  |         for (var i = 0; i < sels.length; i++) | ||||||
|  |           sels[i] = contractSelection(sels[i]); | ||||||
|  |         cm.setSelections(sels); | ||||||
|  |       } else if (type == "both") { | ||||||
|  |         cm.replaceSelection(left + right, null); | ||||||
|  |         cm.triggerElectric(left + right); | ||||||
|  |         cm.execCommand("goCharLeft"); | ||||||
|  |       } else if (type == "addFour") { | ||||||
|  |         cm.replaceSelection(left + left + left + left, "before"); | ||||||
|  |         cm.execCommand("goCharRight"); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function charsAround(cm, pos) { | ||||||
|  |     var str = cm.getRange(Pos(pos.line, pos.ch - 1), | ||||||
|  |                           Pos(pos.line, pos.ch + 1)); | ||||||
|  |     return str.length == 2 ? str : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function stringStartsAfter(cm, pos) { | ||||||
|  |     var token = cm.getTokenAt(Pos(pos.line, pos.ch + 1)) | ||||||
|  |     return /\bstring/.test(token.type) && token.start == pos.ch && | ||||||
|  |       (pos.ch == 0 || !/\bstring/.test(cm.getTokenTypeAt(pos))) | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										184
									
								
								passbook/admin/static/codemirror/addon/edit/closetag.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								passbook/admin/static/codemirror/addon/edit/closetag.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,184 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Tag-closer extension for CodeMirror. | ||||||
|  |  * | ||||||
|  |  * This extension adds an "autoCloseTags" option that can be set to | ||||||
|  |  * either true to get the default behavior, or an object to further | ||||||
|  |  * configure its behavior. | ||||||
|  |  * | ||||||
|  |  * These are supported options: | ||||||
|  |  * | ||||||
|  |  * `whenClosing` (default true) | ||||||
|  |  *   Whether to autoclose when the '/' of a closing tag is typed. | ||||||
|  |  * `whenOpening` (default true) | ||||||
|  |  *   Whether to autoclose the tag when the final '>' of an opening | ||||||
|  |  *   tag is typed. | ||||||
|  |  * `dontCloseTags` (default is empty tags for HTML, none for XML) | ||||||
|  |  *   An array of tag names that should not be autoclosed. | ||||||
|  |  * `indentTags` (default is block tags for HTML, none for XML) | ||||||
|  |  *   An array of tag names that should, when opened, cause a | ||||||
|  |  *   blank line to be added inside the tag, and the blank line and | ||||||
|  |  *   closing line to be indented. | ||||||
|  |  * `emptyTags` (default is none) | ||||||
|  |  *   An array of XML tag names that should be autoclosed with '/>'. | ||||||
|  |  * | ||||||
|  |  * See demos/closetag.html for a usage example. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("../fold/xml-fold")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "../fold/xml-fold"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   CodeMirror.defineOption("autoCloseTags", false, function(cm, val, old) { | ||||||
|  |     if (old != CodeMirror.Init && old) | ||||||
|  |       cm.removeKeyMap("autoCloseTags"); | ||||||
|  |     if (!val) return; | ||||||
|  |     var map = {name: "autoCloseTags"}; | ||||||
|  |     if (typeof val != "object" || val.whenClosing) | ||||||
|  |       map["'/'"] = function(cm) { return autoCloseSlash(cm); }; | ||||||
|  |     if (typeof val != "object" || val.whenOpening) | ||||||
|  |       map["'>'"] = function(cm) { return autoCloseGT(cm); }; | ||||||
|  |     cm.addKeyMap(map); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   var htmlDontClose = ["area", "base", "br", "col", "command", "embed", "hr", "img", "input", "keygen", "link", "meta", "param", | ||||||
|  |                        "source", "track", "wbr"]; | ||||||
|  |   var htmlIndent = ["applet", "blockquote", "body", "button", "div", "dl", "fieldset", "form", "frameset", "h1", "h2", "h3", "h4", | ||||||
|  |                     "h5", "h6", "head", "html", "iframe", "layer", "legend", "object", "ol", "p", "select", "table", "ul"]; | ||||||
|  |  | ||||||
|  |   function autoCloseGT(cm) { | ||||||
|  |     if (cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |     var ranges = cm.listSelections(), replacements = []; | ||||||
|  |     var opt = cm.getOption("autoCloseTags"); | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       if (!ranges[i].empty()) return CodeMirror.Pass; | ||||||
|  |       var pos = ranges[i].head, tok = cm.getTokenAt(pos); | ||||||
|  |       var inner = CodeMirror.innerMode(cm.getMode(), tok.state), state = inner.state; | ||||||
|  |       var tagInfo = inner.mode.xmlCurrentTag && inner.mode.xmlCurrentTag(state) | ||||||
|  |       var tagName = tagInfo && tagInfo.name | ||||||
|  |       if (!tagName) return CodeMirror.Pass | ||||||
|  |  | ||||||
|  |       var html = inner.mode.configuration == "html"; | ||||||
|  |       var dontCloseTags = (typeof opt == "object" && opt.dontCloseTags) || (html && htmlDontClose); | ||||||
|  |       var indentTags = (typeof opt == "object" && opt.indentTags) || (html && htmlIndent); | ||||||
|  |  | ||||||
|  |       if (tok.end > pos.ch) tagName = tagName.slice(0, tagName.length - tok.end + pos.ch); | ||||||
|  |       var lowerTagName = tagName.toLowerCase(); | ||||||
|  |       // Don't process the '>' at the end of an end-tag or self-closing tag | ||||||
|  |       if (!tagName || | ||||||
|  |           tok.type == "string" && (tok.end != pos.ch || !/[\"\']/.test(tok.string.charAt(tok.string.length - 1)) || tok.string.length == 1) || | ||||||
|  |           tok.type == "tag" && tagInfo.close || | ||||||
|  |           tok.string.indexOf("/") == (tok.string.length - 1) || // match something like <someTagName /> | ||||||
|  |           dontCloseTags && indexOf(dontCloseTags, lowerTagName) > -1 || | ||||||
|  |           closingTagExists(cm, inner.mode.xmlCurrentContext && inner.mode.xmlCurrentContext(state) || [], tagName, pos, true)) | ||||||
|  |         return CodeMirror.Pass; | ||||||
|  |  | ||||||
|  |       var emptyTags = typeof opt == "object" && opt.emptyTags; | ||||||
|  |       if (emptyTags && indexOf(emptyTags, tagName) > -1) { | ||||||
|  |         replacements[i] = { text: "/>", newPos: CodeMirror.Pos(pos.line, pos.ch + 2) }; | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var indent = indentTags && indexOf(indentTags, lowerTagName) > -1; | ||||||
|  |       replacements[i] = {indent: indent, | ||||||
|  |                          text: ">" + (indent ? "\n\n" : "") + "</" + tagName + ">", | ||||||
|  |                          newPos: indent ? CodeMirror.Pos(pos.line + 1, 0) : CodeMirror.Pos(pos.line, pos.ch + 1)}; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var dontIndentOnAutoClose = (typeof opt == "object" && opt.dontIndentOnAutoClose); | ||||||
|  |     for (var i = ranges.length - 1; i >= 0; i--) { | ||||||
|  |       var info = replacements[i]; | ||||||
|  |       cm.replaceRange(info.text, ranges[i].head, ranges[i].anchor, "+insert"); | ||||||
|  |       var sel = cm.listSelections().slice(0); | ||||||
|  |       sel[i] = {head: info.newPos, anchor: info.newPos}; | ||||||
|  |       cm.setSelections(sel); | ||||||
|  |       if (!dontIndentOnAutoClose && info.indent) { | ||||||
|  |         cm.indentLine(info.newPos.line, null, true); | ||||||
|  |         cm.indentLine(info.newPos.line + 1, null, true); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function autoCloseCurrent(cm, typingSlash) { | ||||||
|  |     var ranges = cm.listSelections(), replacements = []; | ||||||
|  |     var head = typingSlash ? "/" : "</"; | ||||||
|  |     var opt = cm.getOption("autoCloseTags"); | ||||||
|  |     var dontIndentOnAutoClose = (typeof opt == "object" && opt.dontIndentOnSlash); | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       if (!ranges[i].empty()) return CodeMirror.Pass; | ||||||
|  |       var pos = ranges[i].head, tok = cm.getTokenAt(pos); | ||||||
|  |       var inner = CodeMirror.innerMode(cm.getMode(), tok.state), state = inner.state; | ||||||
|  |       if (typingSlash && (tok.type == "string" || tok.string.charAt(0) != "<" || | ||||||
|  |                           tok.start != pos.ch - 1)) | ||||||
|  |         return CodeMirror.Pass; | ||||||
|  |       // Kludge to get around the fact that we are not in XML mode | ||||||
|  |       // when completing in JS/CSS snippet in htmlmixed mode. Does not | ||||||
|  |       // work for other XML embedded languages (there is no general | ||||||
|  |       // way to go from a mixed mode to its current XML state). | ||||||
|  |       var replacement, mixed = inner.mode.name != "xml" && cm.getMode().name == "htmlmixed" | ||||||
|  |       if (mixed && inner.mode.name == "javascript") { | ||||||
|  |         replacement = head + "script"; | ||||||
|  |       } else if (mixed && inner.mode.name == "css") { | ||||||
|  |         replacement = head + "style"; | ||||||
|  |       } else { | ||||||
|  |         var context = inner.mode.xmlCurrentContext && inner.mode.xmlCurrentContext(state) | ||||||
|  |         if (!context || (context.length && closingTagExists(cm, context, context[context.length - 1], pos))) | ||||||
|  |           return CodeMirror.Pass; | ||||||
|  |         replacement = head + context[context.length - 1] | ||||||
|  |       } | ||||||
|  |       if (cm.getLine(pos.line).charAt(tok.end) != ">") replacement += ">"; | ||||||
|  |       replacements[i] = replacement; | ||||||
|  |     } | ||||||
|  |     cm.replaceSelections(replacements); | ||||||
|  |     ranges = cm.listSelections(); | ||||||
|  |     if (!dontIndentOnAutoClose) { | ||||||
|  |         for (var i = 0; i < ranges.length; i++) | ||||||
|  |             if (i == ranges.length - 1 || ranges[i].head.line < ranges[i + 1].head.line) | ||||||
|  |                 cm.indentLine(ranges[i].head.line); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function autoCloseSlash(cm) { | ||||||
|  |     if (cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |     return autoCloseCurrent(cm, true); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.closeTag = function(cm) { return autoCloseCurrent(cm); }; | ||||||
|  |  | ||||||
|  |   function indexOf(collection, elt) { | ||||||
|  |     if (collection.indexOf) return collection.indexOf(elt); | ||||||
|  |     for (var i = 0, e = collection.length; i < e; ++i) | ||||||
|  |       if (collection[i] == elt) return i; | ||||||
|  |     return -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // If xml-fold is loaded, we use its functionality to try and verify | ||||||
|  |   // whether a given tag is actually unclosed. | ||||||
|  |   function closingTagExists(cm, context, tagName, pos, newTag) { | ||||||
|  |     if (!CodeMirror.scanForClosingTag) return false; | ||||||
|  |     var end = Math.min(cm.lastLine() + 1, pos.line + 500); | ||||||
|  |     var nextClose = CodeMirror.scanForClosingTag(cm, pos, null, end); | ||||||
|  |     if (!nextClose || nextClose.tag != tagName) return false; | ||||||
|  |     // If the immediate wrapping context contains onCx instances of | ||||||
|  |     // the same tag, a closing tag only exists if there are at least | ||||||
|  |     // that many closing tags of that type following. | ||||||
|  |     var onCx = newTag ? 1 : 0 | ||||||
|  |     for (var i = context.length - 1; i >= 0; i--) { | ||||||
|  |       if (context[i] == tagName) ++onCx | ||||||
|  |       else break | ||||||
|  |     } | ||||||
|  |     pos = nextClose.to; | ||||||
|  |     for (var i = 1; i < onCx; i++) { | ||||||
|  |       var next = CodeMirror.scanForClosingTag(cm, pos, null, end); | ||||||
|  |       if (!next || next.tag != tagName) return false; | ||||||
|  |       pos = next.to; | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										99
									
								
								passbook/admin/static/codemirror/addon/edit/continuelist.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								passbook/admin/static/codemirror/addon/edit/continuelist.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,99 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var listRE = /^(\s*)(>[> ]*|[*+-] \[[x ]\]\s|[*+-]\s|(\d+)([.)]))(\s*)/, | ||||||
|  |       emptyListRE = /^(\s*)(>[> ]*|[*+-] \[[x ]\]|[*+-]|(\d+)[.)])(\s*)$/, | ||||||
|  |       unorderedListRE = /[*+-]\s/; | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.newlineAndIndentContinueMarkdownList = function(cm) { | ||||||
|  |     if (cm.getOption("disableInput")) return CodeMirror.Pass; | ||||||
|  |     var ranges = cm.listSelections(), replacements = []; | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       var pos = ranges[i].head; | ||||||
|  |  | ||||||
|  |       // If we're not in Markdown mode, fall back to normal newlineAndIndent | ||||||
|  |       var eolState = cm.getStateAfter(pos.line); | ||||||
|  |       var inner = CodeMirror.innerMode(cm.getMode(), eolState); | ||||||
|  |       if (inner.mode.name !== "markdown") { | ||||||
|  |         cm.execCommand("newlineAndIndent"); | ||||||
|  |         return; | ||||||
|  |       } else { | ||||||
|  |         eolState = inner.state; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var inList = eolState.list !== false; | ||||||
|  |       var inQuote = eolState.quote !== 0; | ||||||
|  |  | ||||||
|  |       var line = cm.getLine(pos.line), match = listRE.exec(line); | ||||||
|  |       var cursorBeforeBullet = /^\s*$/.test(line.slice(0, pos.ch)); | ||||||
|  |       if (!ranges[i].empty() || (!inList && !inQuote) || !match || cursorBeforeBullet) { | ||||||
|  |         cm.execCommand("newlineAndIndent"); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       if (emptyListRE.test(line)) { | ||||||
|  |         if (!/>\s*$/.test(line)) cm.replaceRange("", { | ||||||
|  |           line: pos.line, ch: 0 | ||||||
|  |         }, { | ||||||
|  |           line: pos.line, ch: pos.ch + 1 | ||||||
|  |         }); | ||||||
|  |         replacements[i] = "\n"; | ||||||
|  |       } else { | ||||||
|  |         var indent = match[1], after = match[5]; | ||||||
|  |         var numbered = !(unorderedListRE.test(match[2]) || match[2].indexOf(">") >= 0); | ||||||
|  |         var bullet = numbered ? (parseInt(match[3], 10) + 1) + match[4] : match[2].replace("x", " "); | ||||||
|  |         replacements[i] = "\n" + indent + bullet + after; | ||||||
|  |  | ||||||
|  |         if (numbered) incrementRemainingMarkdownListNumbers(cm, pos); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cm.replaceSelections(replacements); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // Auto-updating Markdown list numbers when a new item is added to the | ||||||
|  |   // middle of a list | ||||||
|  |   function incrementRemainingMarkdownListNumbers(cm, pos) { | ||||||
|  |     var startLine = pos.line, lookAhead = 0, skipCount = 0; | ||||||
|  |     var startItem = listRE.exec(cm.getLine(startLine)), startIndent = startItem[1]; | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |       lookAhead += 1; | ||||||
|  |       var nextLineNumber = startLine + lookAhead; | ||||||
|  |       var nextLine = cm.getLine(nextLineNumber), nextItem = listRE.exec(nextLine); | ||||||
|  |  | ||||||
|  |       if (nextItem) { | ||||||
|  |         var nextIndent = nextItem[1]; | ||||||
|  |         var newNumber = (parseInt(startItem[3], 10) + lookAhead - skipCount); | ||||||
|  |         var nextNumber = (parseInt(nextItem[3], 10)), itemNumber = nextNumber; | ||||||
|  |  | ||||||
|  |         if (startIndent === nextIndent && !isNaN(nextNumber)) { | ||||||
|  |           if (newNumber === nextNumber) itemNumber = nextNumber + 1; | ||||||
|  |           if (newNumber > nextNumber) itemNumber = newNumber + 1; | ||||||
|  |           cm.replaceRange( | ||||||
|  |             nextLine.replace(listRE, nextIndent + itemNumber + nextItem[4] + nextItem[5]), | ||||||
|  |           { | ||||||
|  |             line: nextLineNumber, ch: 0 | ||||||
|  |           }, { | ||||||
|  |             line: nextLineNumber, ch: nextLine.length | ||||||
|  |           }); | ||||||
|  |         } else { | ||||||
|  |           if (startIndent.length > nextIndent.length) return; | ||||||
|  |           // This doesn't run if the next line immediatley indents, as it is | ||||||
|  |           // not clear of the users intention (new indented item or same level) | ||||||
|  |           if ((startIndent.length < nextIndent.length) && (lookAhead === 1)) return; | ||||||
|  |           skipCount += 1; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } while (nextItem); | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										150
									
								
								passbook/admin/static/codemirror/addon/edit/matchbrackets.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								passbook/admin/static/codemirror/addon/edit/matchbrackets.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,150 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   var ie_lt8 = /MSIE \d/.test(navigator.userAgent) && | ||||||
|  |     (document.documentMode == null || document.documentMode < 8); | ||||||
|  |  | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   var matching = {"(": ")>", ")": "(<", "[": "]>", "]": "[<", "{": "}>", "}": "{<", "<": ">>", ">": "<<"}; | ||||||
|  |  | ||||||
|  |   function bracketRegex(config) { | ||||||
|  |     return config && config.bracketRegex || /[(){}[\]]/ | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function findMatchingBracket(cm, where, config) { | ||||||
|  |     var line = cm.getLineHandle(where.line), pos = where.ch - 1; | ||||||
|  |     var afterCursor = config && config.afterCursor | ||||||
|  |     if (afterCursor == null) | ||||||
|  |       afterCursor = /(^| )cm-fat-cursor($| )/.test(cm.getWrapperElement().className) | ||||||
|  |     var re = bracketRegex(config) | ||||||
|  |  | ||||||
|  |     // A cursor is defined as between two characters, but in in vim command mode | ||||||
|  |     // (i.e. not insert mode), the cursor is visually represented as a | ||||||
|  |     // highlighted box on top of the 2nd character. Otherwise, we allow matches | ||||||
|  |     // from before or after the cursor. | ||||||
|  |     var match = (!afterCursor && pos >= 0 && re.test(line.text.charAt(pos)) && matching[line.text.charAt(pos)]) || | ||||||
|  |         re.test(line.text.charAt(pos + 1)) && matching[line.text.charAt(++pos)]; | ||||||
|  |     if (!match) return null; | ||||||
|  |     var dir = match.charAt(1) == ">" ? 1 : -1; | ||||||
|  |     if (config && config.strict && (dir > 0) != (pos == where.ch)) return null; | ||||||
|  |     var style = cm.getTokenTypeAt(Pos(where.line, pos + 1)); | ||||||
|  |  | ||||||
|  |     var found = scanForBracket(cm, Pos(where.line, pos + (dir > 0 ? 1 : 0)), dir, style || null, config); | ||||||
|  |     if (found == null) return null; | ||||||
|  |     return {from: Pos(where.line, pos), to: found && found.pos, | ||||||
|  |             match: found && found.ch == match.charAt(0), forward: dir > 0}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // bracketRegex is used to specify which type of bracket to scan | ||||||
|  |   // should be a regexp, e.g. /[[\]]/ | ||||||
|  |   // | ||||||
|  |   // Note: If "where" is on an open bracket, then this bracket is ignored. | ||||||
|  |   // | ||||||
|  |   // Returns false when no bracket was found, null when it reached | ||||||
|  |   // maxScanLines and gave up | ||||||
|  |   function scanForBracket(cm, where, dir, style, config) { | ||||||
|  |     var maxScanLen = (config && config.maxScanLineLength) || 10000; | ||||||
|  |     var maxScanLines = (config && config.maxScanLines) || 1000; | ||||||
|  |  | ||||||
|  |     var stack = []; | ||||||
|  |     var re = bracketRegex(config) | ||||||
|  |     var lineEnd = dir > 0 ? Math.min(where.line + maxScanLines, cm.lastLine() + 1) | ||||||
|  |                           : Math.max(cm.firstLine() - 1, where.line - maxScanLines); | ||||||
|  |     for (var lineNo = where.line; lineNo != lineEnd; lineNo += dir) { | ||||||
|  |       var line = cm.getLine(lineNo); | ||||||
|  |       if (!line) continue; | ||||||
|  |       var pos = dir > 0 ? 0 : line.length - 1, end = dir > 0 ? line.length : -1; | ||||||
|  |       if (line.length > maxScanLen) continue; | ||||||
|  |       if (lineNo == where.line) pos = where.ch - (dir < 0 ? 1 : 0); | ||||||
|  |       for (; pos != end; pos += dir) { | ||||||
|  |         var ch = line.charAt(pos); | ||||||
|  |         if (re.test(ch) && (style === undefined || cm.getTokenTypeAt(Pos(lineNo, pos + 1)) == style)) { | ||||||
|  |           var match = matching[ch]; | ||||||
|  |           if (match && (match.charAt(1) == ">") == (dir > 0)) stack.push(ch); | ||||||
|  |           else if (!stack.length) return {pos: Pos(lineNo, pos), ch: ch}; | ||||||
|  |           else stack.pop(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return lineNo - dir == (dir > 0 ? cm.lastLine() : cm.firstLine()) ? false : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function matchBrackets(cm, autoclear, config) { | ||||||
|  |     // Disable brace matching in long lines, since it'll cause hugely slow updates | ||||||
|  |     var maxHighlightLen = cm.state.matchBrackets.maxHighlightLineLength || 1000; | ||||||
|  |     var marks = [], ranges = cm.listSelections(); | ||||||
|  |     for (var i = 0; i < ranges.length; i++) { | ||||||
|  |       var match = ranges[i].empty() && findMatchingBracket(cm, ranges[i].head, config); | ||||||
|  |       if (match && cm.getLine(match.from.line).length <= maxHighlightLen) { | ||||||
|  |         var style = match.match ? "CodeMirror-matchingbracket" : "CodeMirror-nonmatchingbracket"; | ||||||
|  |         marks.push(cm.markText(match.from, Pos(match.from.line, match.from.ch + 1), {className: style})); | ||||||
|  |         if (match.to && cm.getLine(match.to.line).length <= maxHighlightLen) | ||||||
|  |           marks.push(cm.markText(match.to, Pos(match.to.line, match.to.ch + 1), {className: style})); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (marks.length) { | ||||||
|  |       // Kludge to work around the IE bug from issue #1193, where text | ||||||
|  |       // input stops going to the textare whever this fires. | ||||||
|  |       if (ie_lt8 && cm.state.focused) cm.focus(); | ||||||
|  |  | ||||||
|  |       var clear = function() { | ||||||
|  |         cm.operation(function() { | ||||||
|  |           for (var i = 0; i < marks.length; i++) marks[i].clear(); | ||||||
|  |         }); | ||||||
|  |       }; | ||||||
|  |       if (autoclear) setTimeout(clear, 800); | ||||||
|  |       else return clear; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function doMatchBrackets(cm) { | ||||||
|  |     cm.operation(function() { | ||||||
|  |       if (cm.state.matchBrackets.currentlyHighlighted) { | ||||||
|  |         cm.state.matchBrackets.currentlyHighlighted(); | ||||||
|  |         cm.state.matchBrackets.currentlyHighlighted = null; | ||||||
|  |       } | ||||||
|  |       cm.state.matchBrackets.currentlyHighlighted = matchBrackets(cm, false, cm.state.matchBrackets); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("matchBrackets", false, function(cm, val, old) { | ||||||
|  |     if (old && old != CodeMirror.Init) { | ||||||
|  |       cm.off("cursorActivity", doMatchBrackets); | ||||||
|  |       if (cm.state.matchBrackets && cm.state.matchBrackets.currentlyHighlighted) { | ||||||
|  |         cm.state.matchBrackets.currentlyHighlighted(); | ||||||
|  |         cm.state.matchBrackets.currentlyHighlighted = null; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (val) { | ||||||
|  |       cm.state.matchBrackets = typeof val == "object" ? val : {}; | ||||||
|  |       cm.on("cursorActivity", doMatchBrackets); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("matchBrackets", function() {matchBrackets(this, true);}); | ||||||
|  |   CodeMirror.defineExtension("findMatchingBracket", function(pos, config, oldConfig){ | ||||||
|  |     // Backwards-compatibility kludge | ||||||
|  |     if (oldConfig || typeof config == "boolean") { | ||||||
|  |       if (!oldConfig) { | ||||||
|  |         config = config ? {strict: true} : null | ||||||
|  |       } else { | ||||||
|  |         oldConfig.strict = config | ||||||
|  |         config = oldConfig | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return findMatchingBracket(this, pos, config) | ||||||
|  |   }); | ||||||
|  |   CodeMirror.defineExtension("scanForBracket", function(pos, dir, style, config){ | ||||||
|  |     return scanForBracket(this, pos, dir, style, config); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										66
									
								
								passbook/admin/static/codemirror/addon/edit/matchtags.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								passbook/admin/static/codemirror/addon/edit/matchtags.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("../fold/xml-fold")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "../fold/xml-fold"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("matchTags", false, function(cm, val, old) { | ||||||
|  |     if (old && old != CodeMirror.Init) { | ||||||
|  |       cm.off("cursorActivity", doMatchTags); | ||||||
|  |       cm.off("viewportChange", maybeUpdateMatch); | ||||||
|  |       clear(cm); | ||||||
|  |     } | ||||||
|  |     if (val) { | ||||||
|  |       cm.state.matchBothTags = typeof val == "object" && val.bothTags; | ||||||
|  |       cm.on("cursorActivity", doMatchTags); | ||||||
|  |       cm.on("viewportChange", maybeUpdateMatch); | ||||||
|  |       doMatchTags(cm); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   function clear(cm) { | ||||||
|  |     if (cm.state.tagHit) cm.state.tagHit.clear(); | ||||||
|  |     if (cm.state.tagOther) cm.state.tagOther.clear(); | ||||||
|  |     cm.state.tagHit = cm.state.tagOther = null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function doMatchTags(cm) { | ||||||
|  |     cm.state.failedTagMatch = false; | ||||||
|  |     cm.operation(function() { | ||||||
|  |       clear(cm); | ||||||
|  |       if (cm.somethingSelected()) return; | ||||||
|  |       var cur = cm.getCursor(), range = cm.getViewport(); | ||||||
|  |       range.from = Math.min(range.from, cur.line); range.to = Math.max(cur.line + 1, range.to); | ||||||
|  |       var match = CodeMirror.findMatchingTag(cm, cur, range); | ||||||
|  |       if (!match) return; | ||||||
|  |       if (cm.state.matchBothTags) { | ||||||
|  |         var hit = match.at == "open" ? match.open : match.close; | ||||||
|  |         if (hit) cm.state.tagHit = cm.markText(hit.from, hit.to, {className: "CodeMirror-matchingtag"}); | ||||||
|  |       } | ||||||
|  |       var other = match.at == "close" ? match.open : match.close; | ||||||
|  |       if (other) | ||||||
|  |         cm.state.tagOther = cm.markText(other.from, other.to, {className: "CodeMirror-matchingtag"}); | ||||||
|  |       else | ||||||
|  |         cm.state.failedTagMatch = true; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function maybeUpdateMatch(cm) { | ||||||
|  |     if (cm.state.failedTagMatch) doMatchTags(cm); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.toMatchingTag = function(cm) { | ||||||
|  |     var found = CodeMirror.findMatchingTag(cm, cm.getCursor()); | ||||||
|  |     if (found) { | ||||||
|  |       var other = found.at == "close" ? found.open : found.close; | ||||||
|  |       if (other) cm.extendSelection(other.to, other.from); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | }); | ||||||
							
								
								
									
										27
									
								
								passbook/admin/static/codemirror/addon/edit/trailingspace.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								passbook/admin/static/codemirror/addon/edit/trailingspace.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   CodeMirror.defineOption("showTrailingSpace", false, function(cm, val, prev) { | ||||||
|  |     if (prev == CodeMirror.Init) prev = false; | ||||||
|  |     if (prev && !val) | ||||||
|  |       cm.removeOverlay("trailingspace"); | ||||||
|  |     else if (!prev && val) | ||||||
|  |       cm.addOverlay({ | ||||||
|  |         token: function(stream) { | ||||||
|  |           for (var l = stream.string.length, i = l; i && /\s/.test(stream.string.charAt(i - 1)); --i) {} | ||||||
|  |           if (i > stream.pos) { stream.pos = i; return null; } | ||||||
|  |           stream.pos = l; | ||||||
|  |           return "trailingspace"; | ||||||
|  |         }, | ||||||
|  |         name: "trailingspace" | ||||||
|  |       }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										105
									
								
								passbook/admin/static/codemirror/addon/fold/brace-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								passbook/admin/static/codemirror/addon/fold/brace-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,105 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("fold", "brace", function(cm, start) { | ||||||
|  |   var line = start.line, lineText = cm.getLine(line); | ||||||
|  |   var tokenType; | ||||||
|  |  | ||||||
|  |   function findOpening(openCh) { | ||||||
|  |     for (var at = start.ch, pass = 0;;) { | ||||||
|  |       var found = at <= 0 ? -1 : lineText.lastIndexOf(openCh, at - 1); | ||||||
|  |       if (found == -1) { | ||||||
|  |         if (pass == 1) break; | ||||||
|  |         pass = 1; | ||||||
|  |         at = lineText.length; | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |       if (pass == 1 && found < start.ch) break; | ||||||
|  |       tokenType = cm.getTokenTypeAt(CodeMirror.Pos(line, found + 1)); | ||||||
|  |       if (!/^(comment|string)/.test(tokenType)) return found + 1; | ||||||
|  |       at = found - 1; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var startToken = "{", endToken = "}", startCh = findOpening("{"); | ||||||
|  |   if (startCh == null) { | ||||||
|  |     startToken = "[", endToken = "]"; | ||||||
|  |     startCh = findOpening("["); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if (startCh == null) return; | ||||||
|  |   var count = 1, lastLine = cm.lastLine(), end, endCh; | ||||||
|  |   outer: for (var i = line; i <= lastLine; ++i) { | ||||||
|  |     var text = cm.getLine(i), pos = i == line ? startCh : 0; | ||||||
|  |     for (;;) { | ||||||
|  |       var nextOpen = text.indexOf(startToken, pos), nextClose = text.indexOf(endToken, pos); | ||||||
|  |       if (nextOpen < 0) nextOpen = text.length; | ||||||
|  |       if (nextClose < 0) nextClose = text.length; | ||||||
|  |       pos = Math.min(nextOpen, nextClose); | ||||||
|  |       if (pos == text.length) break; | ||||||
|  |       if (cm.getTokenTypeAt(CodeMirror.Pos(i, pos + 1)) == tokenType) { | ||||||
|  |         if (pos == nextOpen) ++count; | ||||||
|  |         else if (!--count) { end = i; endCh = pos; break outer; } | ||||||
|  |       } | ||||||
|  |       ++pos; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (end == null || line == end) return; | ||||||
|  |   return {from: CodeMirror.Pos(line, startCh), | ||||||
|  |           to: CodeMirror.Pos(end, endCh)}; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("fold", "import", function(cm, start) { | ||||||
|  |   function hasImport(line) { | ||||||
|  |     if (line < cm.firstLine() || line > cm.lastLine()) return null; | ||||||
|  |     var start = cm.getTokenAt(CodeMirror.Pos(line, 1)); | ||||||
|  |     if (!/\S/.test(start.string)) start = cm.getTokenAt(CodeMirror.Pos(line, start.end + 1)); | ||||||
|  |     if (start.type != "keyword" || start.string != "import") return null; | ||||||
|  |     // Now find closing semicolon, return its position | ||||||
|  |     for (var i = line, e = Math.min(cm.lastLine(), line + 10); i <= e; ++i) { | ||||||
|  |       var text = cm.getLine(i), semi = text.indexOf(";"); | ||||||
|  |       if (semi != -1) return {startCh: start.end, end: CodeMirror.Pos(i, semi)}; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var startLine = start.line, has = hasImport(startLine), prev; | ||||||
|  |   if (!has || hasImport(startLine - 1) || ((prev = hasImport(startLine - 2)) && prev.end.line == startLine - 1)) | ||||||
|  |     return null; | ||||||
|  |   for (var end = has.end;;) { | ||||||
|  |     var next = hasImport(end.line + 1); | ||||||
|  |     if (next == null) break; | ||||||
|  |     end = next.end; | ||||||
|  |   } | ||||||
|  |   return {from: cm.clipPos(CodeMirror.Pos(startLine, has.startCh + 1)), to: end}; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("fold", "include", function(cm, start) { | ||||||
|  |   function hasInclude(line) { | ||||||
|  |     if (line < cm.firstLine() || line > cm.lastLine()) return null; | ||||||
|  |     var start = cm.getTokenAt(CodeMirror.Pos(line, 1)); | ||||||
|  |     if (!/\S/.test(start.string)) start = cm.getTokenAt(CodeMirror.Pos(line, start.end + 1)); | ||||||
|  |     if (start.type == "meta" && start.string.slice(0, 8) == "#include") return start.start + 8; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var startLine = start.line, has = hasInclude(startLine); | ||||||
|  |   if (has == null || hasInclude(startLine - 1) != null) return null; | ||||||
|  |   for (var end = startLine;;) { | ||||||
|  |     var next = hasInclude(end + 1); | ||||||
|  |     if (next == null) break; | ||||||
|  |     ++end; | ||||||
|  |   } | ||||||
|  |   return {from: CodeMirror.Pos(startLine, has + 1), | ||||||
|  |           to: cm.clipPos(CodeMirror.Pos(end))}; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										59
									
								
								passbook/admin/static/codemirror/addon/fold/comment-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								passbook/admin/static/codemirror/addon/fold/comment-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | CodeMirror.registerGlobalHelper("fold", "comment", function(mode) { | ||||||
|  |   return mode.blockCommentStart && mode.blockCommentEnd; | ||||||
|  | }, function(cm, start) { | ||||||
|  |   var mode = cm.getModeAt(start), startToken = mode.blockCommentStart, endToken = mode.blockCommentEnd; | ||||||
|  |   if (!startToken || !endToken) return; | ||||||
|  |   var line = start.line, lineText = cm.getLine(line); | ||||||
|  |  | ||||||
|  |   var startCh; | ||||||
|  |   for (var at = start.ch, pass = 0;;) { | ||||||
|  |     var found = at <= 0 ? -1 : lineText.lastIndexOf(startToken, at - 1); | ||||||
|  |     if (found == -1) { | ||||||
|  |       if (pass == 1) return; | ||||||
|  |       pass = 1; | ||||||
|  |       at = lineText.length; | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     if (pass == 1 && found < start.ch) return; | ||||||
|  |     if (/comment/.test(cm.getTokenTypeAt(CodeMirror.Pos(line, found + 1))) && | ||||||
|  |         (found == 0 || lineText.slice(found - endToken.length, found) == endToken || | ||||||
|  |          !/comment/.test(cm.getTokenTypeAt(CodeMirror.Pos(line, found))))) { | ||||||
|  |       startCh = found + startToken.length; | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |     at = found - 1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var depth = 1, lastLine = cm.lastLine(), end, endCh; | ||||||
|  |   outer: for (var i = line; i <= lastLine; ++i) { | ||||||
|  |     var text = cm.getLine(i), pos = i == line ? startCh : 0; | ||||||
|  |     for (;;) { | ||||||
|  |       var nextOpen = text.indexOf(startToken, pos), nextClose = text.indexOf(endToken, pos); | ||||||
|  |       if (nextOpen < 0) nextOpen = text.length; | ||||||
|  |       if (nextClose < 0) nextClose = text.length; | ||||||
|  |       pos = Math.min(nextOpen, nextClose); | ||||||
|  |       if (pos == text.length) break; | ||||||
|  |       if (pos == nextOpen) ++depth; | ||||||
|  |       else if (!--depth) { end = i; endCh = pos; break outer; } | ||||||
|  |       ++pos; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (end == null || line == end && endCh == startCh) return; | ||||||
|  |   return {from: CodeMirror.Pos(line, startCh), | ||||||
|  |           to: CodeMirror.Pos(end, endCh)}; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										152
									
								
								passbook/admin/static/codemirror/addon/fold/foldcode.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										152
									
								
								passbook/admin/static/codemirror/addon/fold/foldcode.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,152 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   function doFold(cm, pos, options, force) { | ||||||
|  |     if (options && options.call) { | ||||||
|  |       var finder = options; | ||||||
|  |       options = null; | ||||||
|  |     } else { | ||||||
|  |       var finder = getOption(cm, options, "rangeFinder"); | ||||||
|  |     } | ||||||
|  |     if (typeof pos == "number") pos = CodeMirror.Pos(pos, 0); | ||||||
|  |     var minSize = getOption(cm, options, "minFoldSize"); | ||||||
|  |  | ||||||
|  |     function getRange(allowFolded) { | ||||||
|  |       var range = finder(cm, pos); | ||||||
|  |       if (!range || range.to.line - range.from.line < minSize) return null; | ||||||
|  |       var marks = cm.findMarksAt(range.from); | ||||||
|  |       for (var i = 0; i < marks.length; ++i) { | ||||||
|  |         if (marks[i].__isFold && force !== "fold") { | ||||||
|  |           if (!allowFolded) return null; | ||||||
|  |           range.cleared = true; | ||||||
|  |           marks[i].clear(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return range; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var range = getRange(true); | ||||||
|  |     if (getOption(cm, options, "scanUp")) while (!range && pos.line > cm.firstLine()) { | ||||||
|  |       pos = CodeMirror.Pos(pos.line - 1, 0); | ||||||
|  |       range = getRange(false); | ||||||
|  |     } | ||||||
|  |     if (!range || range.cleared || force === "unfold") return; | ||||||
|  |  | ||||||
|  |     var myWidget = makeWidget(cm, options); | ||||||
|  |     CodeMirror.on(myWidget, "mousedown", function(e) { | ||||||
|  |       myRange.clear(); | ||||||
|  |       CodeMirror.e_preventDefault(e); | ||||||
|  |     }); | ||||||
|  |     var myRange = cm.markText(range.from, range.to, { | ||||||
|  |       replacedWith: myWidget, | ||||||
|  |       clearOnEnter: getOption(cm, options, "clearOnEnter"), | ||||||
|  |       __isFold: true | ||||||
|  |     }); | ||||||
|  |     myRange.on("clear", function(from, to) { | ||||||
|  |       CodeMirror.signal(cm, "unfold", cm, from, to); | ||||||
|  |     }); | ||||||
|  |     CodeMirror.signal(cm, "fold", cm, range.from, range.to); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function makeWidget(cm, options) { | ||||||
|  |     var widget = getOption(cm, options, "widget"); | ||||||
|  |     if (typeof widget == "string") { | ||||||
|  |       var text = document.createTextNode(widget); | ||||||
|  |       widget = document.createElement("span"); | ||||||
|  |       widget.appendChild(text); | ||||||
|  |       widget.className = "CodeMirror-foldmarker"; | ||||||
|  |     } else if (widget) { | ||||||
|  |       widget = widget.cloneNode(true) | ||||||
|  |     } | ||||||
|  |     return widget; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Clumsy backwards-compatible interface | ||||||
|  |   CodeMirror.newFoldFunction = function(rangeFinder, widget) { | ||||||
|  |     return function(cm, pos) { doFold(cm, pos, {rangeFinder: rangeFinder, widget: widget}); }; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // New-style interface | ||||||
|  |   CodeMirror.defineExtension("foldCode", function(pos, options, force) { | ||||||
|  |     doFold(this, pos, options, force); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("isFolded", function(pos) { | ||||||
|  |     var marks = this.findMarksAt(pos); | ||||||
|  |     for (var i = 0; i < marks.length; ++i) | ||||||
|  |       if (marks[i].__isFold) return true; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.toggleFold = function(cm) { | ||||||
|  |     cm.foldCode(cm.getCursor()); | ||||||
|  |   }; | ||||||
|  |   CodeMirror.commands.fold = function(cm) { | ||||||
|  |     cm.foldCode(cm.getCursor(), null, "fold"); | ||||||
|  |   }; | ||||||
|  |   CodeMirror.commands.unfold = function(cm) { | ||||||
|  |     cm.foldCode(cm.getCursor(), null, "unfold"); | ||||||
|  |   }; | ||||||
|  |   CodeMirror.commands.foldAll = function(cm) { | ||||||
|  |     cm.operation(function() { | ||||||
|  |       for (var i = cm.firstLine(), e = cm.lastLine(); i <= e; i++) | ||||||
|  |         cm.foldCode(CodeMirror.Pos(i, 0), null, "fold"); | ||||||
|  |     }); | ||||||
|  |   }; | ||||||
|  |   CodeMirror.commands.unfoldAll = function(cm) { | ||||||
|  |     cm.operation(function() { | ||||||
|  |       for (var i = cm.firstLine(), e = cm.lastLine(); i <= e; i++) | ||||||
|  |         cm.foldCode(CodeMirror.Pos(i, 0), null, "unfold"); | ||||||
|  |     }); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("fold", "combine", function() { | ||||||
|  |     var funcs = Array.prototype.slice.call(arguments, 0); | ||||||
|  |     return function(cm, start) { | ||||||
|  |       for (var i = 0; i < funcs.length; ++i) { | ||||||
|  |         var found = funcs[i](cm, start); | ||||||
|  |         if (found) return found; | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("fold", "auto", function(cm, start) { | ||||||
|  |     var helpers = cm.getHelpers(start, "fold"); | ||||||
|  |     for (var i = 0; i < helpers.length; i++) { | ||||||
|  |       var cur = helpers[i](cm, start); | ||||||
|  |       if (cur) return cur; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   var defaultOptions = { | ||||||
|  |     rangeFinder: CodeMirror.fold.auto, | ||||||
|  |     widget: "\u2194", | ||||||
|  |     minFoldSize: 0, | ||||||
|  |     scanUp: false, | ||||||
|  |     clearOnEnter: true | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("foldOptions", null); | ||||||
|  |  | ||||||
|  |   function getOption(cm, options, name) { | ||||||
|  |     if (options && options[name] !== undefined) | ||||||
|  |       return options[name]; | ||||||
|  |     var editorOptions = cm.options.foldOptions; | ||||||
|  |     if (editorOptions && editorOptions[name] !== undefined) | ||||||
|  |       return editorOptions[name]; | ||||||
|  |     return defaultOptions[name]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("foldOption", function(options, name) { | ||||||
|  |     return getOption(this, options, name); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										20
									
								
								passbook/admin/static/codemirror/addon/fold/foldgutter.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								passbook/admin/static/codemirror/addon/fold/foldgutter.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | .CodeMirror-foldmarker { | ||||||
|  |   color: blue; | ||||||
|  |   text-shadow: #b9f 1px 1px 2px, #b9f -1px -1px 2px, #b9f 1px -1px 2px, #b9f -1px 1px 2px; | ||||||
|  |   font-family: arial; | ||||||
|  |   line-height: .3; | ||||||
|  |   cursor: pointer; | ||||||
|  | } | ||||||
|  | .CodeMirror-foldgutter { | ||||||
|  |   width: .7em; | ||||||
|  | } | ||||||
|  | .CodeMirror-foldgutter-open, | ||||||
|  | .CodeMirror-foldgutter-folded { | ||||||
|  |   cursor: pointer; | ||||||
|  | } | ||||||
|  | .CodeMirror-foldgutter-open:after { | ||||||
|  |   content: "\25BE"; | ||||||
|  | } | ||||||
|  | .CodeMirror-foldgutter-folded:after { | ||||||
|  |   content: "\25B8"; | ||||||
|  | } | ||||||
							
								
								
									
										151
									
								
								passbook/admin/static/codemirror/addon/fold/foldgutter.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								passbook/admin/static/codemirror/addon/fold/foldgutter.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,151 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("./foldcode")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "./foldcode"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("foldGutter", false, function(cm, val, old) { | ||||||
|  |     if (old && old != CodeMirror.Init) { | ||||||
|  |       cm.clearGutter(cm.state.foldGutter.options.gutter); | ||||||
|  |       cm.state.foldGutter = null; | ||||||
|  |       cm.off("gutterClick", onGutterClick); | ||||||
|  |       cm.off("changes", onChange); | ||||||
|  |       cm.off("viewportChange", onViewportChange); | ||||||
|  |       cm.off("fold", onFold); | ||||||
|  |       cm.off("unfold", onFold); | ||||||
|  |       cm.off("swapDoc", onChange); | ||||||
|  |     } | ||||||
|  |     if (val) { | ||||||
|  |       cm.state.foldGutter = new State(parseOptions(val)); | ||||||
|  |       updateInViewport(cm); | ||||||
|  |       cm.on("gutterClick", onGutterClick); | ||||||
|  |       cm.on("changes", onChange); | ||||||
|  |       cm.on("viewportChange", onViewportChange); | ||||||
|  |       cm.on("fold", onFold); | ||||||
|  |       cm.on("unfold", onFold); | ||||||
|  |       cm.on("swapDoc", onChange); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   function State(options) { | ||||||
|  |     this.options = options; | ||||||
|  |     this.from = this.to = 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function parseOptions(opts) { | ||||||
|  |     if (opts === true) opts = {}; | ||||||
|  |     if (opts.gutter == null) opts.gutter = "CodeMirror-foldgutter"; | ||||||
|  |     if (opts.indicatorOpen == null) opts.indicatorOpen = "CodeMirror-foldgutter-open"; | ||||||
|  |     if (opts.indicatorFolded == null) opts.indicatorFolded = "CodeMirror-foldgutter-folded"; | ||||||
|  |     return opts; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function isFolded(cm, line) { | ||||||
|  |     var marks = cm.findMarks(Pos(line, 0), Pos(line + 1, 0)); | ||||||
|  |     for (var i = 0; i < marks.length; ++i) { | ||||||
|  |       if (marks[i].__isFold) { | ||||||
|  |         var fromPos = marks[i].find(-1); | ||||||
|  |         if (fromPos && fromPos.line === line) | ||||||
|  |           return marks[i]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function marker(spec) { | ||||||
|  |     if (typeof spec == "string") { | ||||||
|  |       var elt = document.createElement("div"); | ||||||
|  |       elt.className = spec + " CodeMirror-guttermarker-subtle"; | ||||||
|  |       return elt; | ||||||
|  |     } else { | ||||||
|  |       return spec.cloneNode(true); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function updateFoldInfo(cm, from, to) { | ||||||
|  |     var opts = cm.state.foldGutter.options, cur = from; | ||||||
|  |     var minSize = cm.foldOption(opts, "minFoldSize"); | ||||||
|  |     var func = cm.foldOption(opts, "rangeFinder"); | ||||||
|  |     cm.eachLine(from, to, function(line) { | ||||||
|  |       var mark = null; | ||||||
|  |       if (isFolded(cm, cur)) { | ||||||
|  |         mark = marker(opts.indicatorFolded); | ||||||
|  |       } else { | ||||||
|  |         var pos = Pos(cur, 0); | ||||||
|  |         var range = func && func(cm, pos); | ||||||
|  |         if (range && range.to.line - range.from.line >= minSize) | ||||||
|  |           mark = marker(opts.indicatorOpen); | ||||||
|  |       } | ||||||
|  |       cm.setGutterMarker(line, opts.gutter, mark); | ||||||
|  |       ++cur; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function updateInViewport(cm) { | ||||||
|  |     var vp = cm.getViewport(), state = cm.state.foldGutter; | ||||||
|  |     if (!state) return; | ||||||
|  |     cm.operation(function() { | ||||||
|  |       updateFoldInfo(cm, vp.from, vp.to); | ||||||
|  |     }); | ||||||
|  |     state.from = vp.from; state.to = vp.to; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function onGutterClick(cm, line, gutter) { | ||||||
|  |     var state = cm.state.foldGutter; | ||||||
|  |     if (!state) return; | ||||||
|  |     var opts = state.options; | ||||||
|  |     if (gutter != opts.gutter) return; | ||||||
|  |     var folded = isFolded(cm, line); | ||||||
|  |     if (folded) folded.clear(); | ||||||
|  |     else cm.foldCode(Pos(line, 0), opts); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function onChange(cm) { | ||||||
|  |     var state = cm.state.foldGutter; | ||||||
|  |     if (!state) return; | ||||||
|  |     var opts = state.options; | ||||||
|  |     state.from = state.to = 0; | ||||||
|  |     clearTimeout(state.changeUpdate); | ||||||
|  |     state.changeUpdate = setTimeout(function() { updateInViewport(cm); }, opts.foldOnChangeTimeSpan || 600); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function onViewportChange(cm) { | ||||||
|  |     var state = cm.state.foldGutter; | ||||||
|  |     if (!state) return; | ||||||
|  |     var opts = state.options; | ||||||
|  |     clearTimeout(state.changeUpdate); | ||||||
|  |     state.changeUpdate = setTimeout(function() { | ||||||
|  |       var vp = cm.getViewport(); | ||||||
|  |       if (state.from == state.to || vp.from - state.to > 20 || state.from - vp.to > 20) { | ||||||
|  |         updateInViewport(cm); | ||||||
|  |       } else { | ||||||
|  |         cm.operation(function() { | ||||||
|  |           if (vp.from < state.from) { | ||||||
|  |             updateFoldInfo(cm, vp.from, state.from); | ||||||
|  |             state.from = vp.from; | ||||||
|  |           } | ||||||
|  |           if (vp.to > state.to) { | ||||||
|  |             updateFoldInfo(cm, state.to, vp.to); | ||||||
|  |             state.to = vp.to; | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |     }, opts.updateViewportTimeSpan || 400); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function onFold(cm, from) { | ||||||
|  |     var state = cm.state.foldGutter; | ||||||
|  |     if (!state) return; | ||||||
|  |     var line = from.line; | ||||||
|  |     if (line >= state.from && line < state.to) | ||||||
|  |       updateFoldInfo(cm, line, line + 1); | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										48
									
								
								passbook/admin/static/codemirror/addon/fold/indent-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								passbook/admin/static/codemirror/addon/fold/indent-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | function lineIndent(cm, lineNo) { | ||||||
|  |   var text = cm.getLine(lineNo) | ||||||
|  |   var spaceTo = text.search(/\S/) | ||||||
|  |   if (spaceTo == -1 || /\bcomment\b/.test(cm.getTokenTypeAt(CodeMirror.Pos(lineNo, spaceTo + 1)))) | ||||||
|  |     return -1 | ||||||
|  |   return CodeMirror.countColumn(text, null, cm.getOption("tabSize")) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("fold", "indent", function(cm, start) { | ||||||
|  |   var myIndent = lineIndent(cm, start.line) | ||||||
|  |   if (myIndent < 0) return | ||||||
|  |   var lastLineInFold = null | ||||||
|  |  | ||||||
|  |   // Go through lines until we find a line that definitely doesn't belong in | ||||||
|  |   // the block we're folding, or to the end. | ||||||
|  |   for (var i = start.line + 1, end = cm.lastLine(); i <= end; ++i) { | ||||||
|  |     var indent = lineIndent(cm, i) | ||||||
|  |     if (indent == -1) { | ||||||
|  |     } else if (indent > myIndent) { | ||||||
|  |       // Lines with a greater indent are considered part of the block. | ||||||
|  |       lastLineInFold = i; | ||||||
|  |     } else { | ||||||
|  |       // If this line has non-space, non-comment content, and is | ||||||
|  |       // indented less or equal to the start line, it is the start of | ||||||
|  |       // another block. | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (lastLineInFold) return { | ||||||
|  |     from: CodeMirror.Pos(start.line, cm.getLine(start.line).length), | ||||||
|  |     to: CodeMirror.Pos(lastLineInFold, cm.getLine(lastLineInFold).length) | ||||||
|  |   }; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										49
									
								
								passbook/admin/static/codemirror/addon/fold/markdown-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								passbook/admin/static/codemirror/addon/fold/markdown-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,49 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("fold", "markdown", function(cm, start) { | ||||||
|  |   var maxDepth = 100; | ||||||
|  |  | ||||||
|  |   function isHeader(lineNo) { | ||||||
|  |     var tokentype = cm.getTokenTypeAt(CodeMirror.Pos(lineNo, 0)); | ||||||
|  |     return tokentype && /\bheader\b/.test(tokentype); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function headerLevel(lineNo, line, nextLine) { | ||||||
|  |     var match = line && line.match(/^#+/); | ||||||
|  |     if (match && isHeader(lineNo)) return match[0].length; | ||||||
|  |     match = nextLine && nextLine.match(/^[=\-]+\s*$/); | ||||||
|  |     if (match && isHeader(lineNo + 1)) return nextLine[0] == "=" ? 1 : 2; | ||||||
|  |     return maxDepth; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var firstLine = cm.getLine(start.line), nextLine = cm.getLine(start.line + 1); | ||||||
|  |   var level = headerLevel(start.line, firstLine, nextLine); | ||||||
|  |   if (level === maxDepth) return undefined; | ||||||
|  |  | ||||||
|  |   var lastLineNo = cm.lastLine(); | ||||||
|  |   var end = start.line, nextNextLine = cm.getLine(end + 2); | ||||||
|  |   while (end < lastLineNo) { | ||||||
|  |     if (headerLevel(end + 1, nextLine, nextNextLine) <= level) break; | ||||||
|  |     ++end; | ||||||
|  |     nextLine = nextNextLine; | ||||||
|  |     nextNextLine = cm.getLine(end + 2); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     from: CodeMirror.Pos(start.line, firstLine.length), | ||||||
|  |     to: CodeMirror.Pos(end, cm.getLine(end).length) | ||||||
|  |   }; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										184
									
								
								passbook/admin/static/codemirror/addon/fold/xml-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								passbook/admin/static/codemirror/addon/fold/xml-fold.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,184 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |   function cmp(a, b) { return a.line - b.line || a.ch - b.ch; } | ||||||
|  |  | ||||||
|  |   var nameStartChar = "A-Z_a-z\\u00C0-\\u00D6\\u00D8-\\u00F6\\u00F8-\\u02FF\\u0370-\\u037D\\u037F-\\u1FFF\\u200C-\\u200D\\u2070-\\u218F\\u2C00-\\u2FEF\\u3001-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFFD"; | ||||||
|  |   var nameChar = nameStartChar + "\-\:\.0-9\\u00B7\\u0300-\\u036F\\u203F-\\u2040"; | ||||||
|  |   var xmlTagStart = new RegExp("<(/?)([" + nameStartChar + "][" + nameChar + "]*)", "g"); | ||||||
|  |  | ||||||
|  |   function Iter(cm, line, ch, range) { | ||||||
|  |     this.line = line; this.ch = ch; | ||||||
|  |     this.cm = cm; this.text = cm.getLine(line); | ||||||
|  |     this.min = range ? Math.max(range.from, cm.firstLine()) : cm.firstLine(); | ||||||
|  |     this.max = range ? Math.min(range.to - 1, cm.lastLine()) : cm.lastLine(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function tagAt(iter, ch) { | ||||||
|  |     var type = iter.cm.getTokenTypeAt(Pos(iter.line, ch)); | ||||||
|  |     return type && /\btag\b/.test(type); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function nextLine(iter) { | ||||||
|  |     if (iter.line >= iter.max) return; | ||||||
|  |     iter.ch = 0; | ||||||
|  |     iter.text = iter.cm.getLine(++iter.line); | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |   function prevLine(iter) { | ||||||
|  |     if (iter.line <= iter.min) return; | ||||||
|  |     iter.text = iter.cm.getLine(--iter.line); | ||||||
|  |     iter.ch = iter.text.length; | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function toTagEnd(iter) { | ||||||
|  |     for (;;) { | ||||||
|  |       var gt = iter.text.indexOf(">", iter.ch); | ||||||
|  |       if (gt == -1) { if (nextLine(iter)) continue; else return; } | ||||||
|  |       if (!tagAt(iter, gt + 1)) { iter.ch = gt + 1; continue; } | ||||||
|  |       var lastSlash = iter.text.lastIndexOf("/", gt); | ||||||
|  |       var selfClose = lastSlash > -1 && !/\S/.test(iter.text.slice(lastSlash + 1, gt)); | ||||||
|  |       iter.ch = gt + 1; | ||||||
|  |       return selfClose ? "selfClose" : "regular"; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function toTagStart(iter) { | ||||||
|  |     for (;;) { | ||||||
|  |       var lt = iter.ch ? iter.text.lastIndexOf("<", iter.ch - 1) : -1; | ||||||
|  |       if (lt == -1) { if (prevLine(iter)) continue; else return; } | ||||||
|  |       if (!tagAt(iter, lt + 1)) { iter.ch = lt; continue; } | ||||||
|  |       xmlTagStart.lastIndex = lt; | ||||||
|  |       iter.ch = lt; | ||||||
|  |       var match = xmlTagStart.exec(iter.text); | ||||||
|  |       if (match && match.index == lt) return match; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function toNextTag(iter) { | ||||||
|  |     for (;;) { | ||||||
|  |       xmlTagStart.lastIndex = iter.ch; | ||||||
|  |       var found = xmlTagStart.exec(iter.text); | ||||||
|  |       if (!found) { if (nextLine(iter)) continue; else return; } | ||||||
|  |       if (!tagAt(iter, found.index + 1)) { iter.ch = found.index + 1; continue; } | ||||||
|  |       iter.ch = found.index + found[0].length; | ||||||
|  |       return found; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function toPrevTag(iter) { | ||||||
|  |     for (;;) { | ||||||
|  |       var gt = iter.ch ? iter.text.lastIndexOf(">", iter.ch - 1) : -1; | ||||||
|  |       if (gt == -1) { if (prevLine(iter)) continue; else return; } | ||||||
|  |       if (!tagAt(iter, gt + 1)) { iter.ch = gt; continue; } | ||||||
|  |       var lastSlash = iter.text.lastIndexOf("/", gt); | ||||||
|  |       var selfClose = lastSlash > -1 && !/\S/.test(iter.text.slice(lastSlash + 1, gt)); | ||||||
|  |       iter.ch = gt + 1; | ||||||
|  |       return selfClose ? "selfClose" : "regular"; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function findMatchingClose(iter, tag) { | ||||||
|  |     var stack = []; | ||||||
|  |     for (;;) { | ||||||
|  |       var next = toNextTag(iter), end, startLine = iter.line, startCh = iter.ch - (next ? next[0].length : 0); | ||||||
|  |       if (!next || !(end = toTagEnd(iter))) return; | ||||||
|  |       if (end == "selfClose") continue; | ||||||
|  |       if (next[1]) { // closing tag | ||||||
|  |         for (var i = stack.length - 1; i >= 0; --i) if (stack[i] == next[2]) { | ||||||
|  |           stack.length = i; | ||||||
|  |           break; | ||||||
|  |         } | ||||||
|  |         if (i < 0 && (!tag || tag == next[2])) return { | ||||||
|  |           tag: next[2], | ||||||
|  |           from: Pos(startLine, startCh), | ||||||
|  |           to: Pos(iter.line, iter.ch) | ||||||
|  |         }; | ||||||
|  |       } else { // opening tag | ||||||
|  |         stack.push(next[2]); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   function findMatchingOpen(iter, tag) { | ||||||
|  |     var stack = []; | ||||||
|  |     for (;;) { | ||||||
|  |       var prev = toPrevTag(iter); | ||||||
|  |       if (!prev) return; | ||||||
|  |       if (prev == "selfClose") { toTagStart(iter); continue; } | ||||||
|  |       var endLine = iter.line, endCh = iter.ch; | ||||||
|  |       var start = toTagStart(iter); | ||||||
|  |       if (!start) return; | ||||||
|  |       if (start[1]) { // closing tag | ||||||
|  |         stack.push(start[2]); | ||||||
|  |       } else { // opening tag | ||||||
|  |         for (var i = stack.length - 1; i >= 0; --i) if (stack[i] == start[2]) { | ||||||
|  |           stack.length = i; | ||||||
|  |           break; | ||||||
|  |         } | ||||||
|  |         if (i < 0 && (!tag || tag == start[2])) return { | ||||||
|  |           tag: start[2], | ||||||
|  |           from: Pos(iter.line, iter.ch), | ||||||
|  |           to: Pos(endLine, endCh) | ||||||
|  |         }; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("fold", "xml", function(cm, start) { | ||||||
|  |     var iter = new Iter(cm, start.line, 0); | ||||||
|  |     for (;;) { | ||||||
|  |       var openTag = toNextTag(iter) | ||||||
|  |       if (!openTag || iter.line != start.line) return | ||||||
|  |       var end = toTagEnd(iter) | ||||||
|  |       if (!end) return | ||||||
|  |       if (!openTag[1] && end != "selfClose") { | ||||||
|  |         var startPos = Pos(iter.line, iter.ch); | ||||||
|  |         var endPos = findMatchingClose(iter, openTag[2]); | ||||||
|  |         return endPos && cmp(endPos.from, startPos) > 0 ? {from: startPos, to: endPos.from} : null | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   CodeMirror.findMatchingTag = function(cm, pos, range) { | ||||||
|  |     var iter = new Iter(cm, pos.line, pos.ch, range); | ||||||
|  |     if (iter.text.indexOf(">") == -1 && iter.text.indexOf("<") == -1) return; | ||||||
|  |     var end = toTagEnd(iter), to = end && Pos(iter.line, iter.ch); | ||||||
|  |     var start = end && toTagStart(iter); | ||||||
|  |     if (!end || !start || cmp(iter, pos) > 0) return; | ||||||
|  |     var here = {from: Pos(iter.line, iter.ch), to: to, tag: start[2]}; | ||||||
|  |     if (end == "selfClose") return {open: here, close: null, at: "open"}; | ||||||
|  |  | ||||||
|  |     if (start[1]) { // closing tag | ||||||
|  |       return {open: findMatchingOpen(iter, start[2]), close: here, at: "close"}; | ||||||
|  |     } else { // opening tag | ||||||
|  |       iter = new Iter(cm, to.line, to.ch, range); | ||||||
|  |       return {open: here, close: findMatchingClose(iter, start[2]), at: "open"}; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.findEnclosingTag = function(cm, pos, range, tag) { | ||||||
|  |     var iter = new Iter(cm, pos.line, pos.ch, range); | ||||||
|  |     for (;;) { | ||||||
|  |       var open = findMatchingOpen(iter, tag); | ||||||
|  |       if (!open) break; | ||||||
|  |       var forward = new Iter(cm, pos.line, pos.ch, range); | ||||||
|  |       var close = findMatchingClose(forward, open.tag); | ||||||
|  |       if (close) return {open: open, close: close}; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // Used by addon/edit/closetag.js | ||||||
|  |   CodeMirror.scanForClosingTag = function(cm, pos, name, end) { | ||||||
|  |     var iter = new Iter(cm, pos.line, pos.ch, end ? {from: 0, to: end} : null); | ||||||
|  |     return findMatchingClose(iter, name); | ||||||
|  |   }; | ||||||
|  | }); | ||||||
							
								
								
									
										41
									
								
								passbook/admin/static/codemirror/addon/hint/anyword-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								passbook/admin/static/codemirror/addon/hint/anyword-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var WORD = /[\w$]+/, RANGE = 500; | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "anyword", function(editor, options) { | ||||||
|  |     var word = options && options.word || WORD; | ||||||
|  |     var range = options && options.range || RANGE; | ||||||
|  |     var cur = editor.getCursor(), curLine = editor.getLine(cur.line); | ||||||
|  |     var end = cur.ch, start = end; | ||||||
|  |     while (start && word.test(curLine.charAt(start - 1))) --start; | ||||||
|  |     var curWord = start != end && curLine.slice(start, end); | ||||||
|  |  | ||||||
|  |     var list = options && options.list || [], seen = {}; | ||||||
|  |     var re = new RegExp(word.source, "g"); | ||||||
|  |     for (var dir = -1; dir <= 1; dir += 2) { | ||||||
|  |       var line = cur.line, endLine = Math.min(Math.max(line + dir * range, editor.firstLine()), editor.lastLine()) + dir; | ||||||
|  |       for (; line != endLine; line += dir) { | ||||||
|  |         var text = editor.getLine(line), m; | ||||||
|  |         while (m = re.exec(text)) { | ||||||
|  |           if (line == cur.line && m[0] === curWord) continue; | ||||||
|  |           if ((!curWord || m[0].lastIndexOf(curWord, 0) == 0) && !Object.prototype.hasOwnProperty.call(seen, m[0])) { | ||||||
|  |             seen[m[0]] = true; | ||||||
|  |             list.push(m[0]); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return {list: list, from: CodeMirror.Pos(cur.line, start), to: CodeMirror.Pos(cur.line, end)}; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										60
									
								
								passbook/admin/static/codemirror/addon/hint/css-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								passbook/admin/static/codemirror/addon/hint/css-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,60 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("../../mode/css/css")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "../../mode/css/css"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var pseudoClasses = {link: 1, visited: 1, active: 1, hover: 1, focus: 1, | ||||||
|  |                        "first-letter": 1, "first-line": 1, "first-child": 1, | ||||||
|  |                        before: 1, after: 1, lang: 1}; | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "css", function(cm) { | ||||||
|  |     var cur = cm.getCursor(), token = cm.getTokenAt(cur); | ||||||
|  |     var inner = CodeMirror.innerMode(cm.getMode(), token.state); | ||||||
|  |     if (inner.mode.name != "css") return; | ||||||
|  |  | ||||||
|  |     if (token.type == "keyword" && "!important".indexOf(token.string) == 0) | ||||||
|  |       return {list: ["!important"], from: CodeMirror.Pos(cur.line, token.start), | ||||||
|  |               to: CodeMirror.Pos(cur.line, token.end)}; | ||||||
|  |  | ||||||
|  |     var start = token.start, end = cur.ch, word = token.string.slice(0, end - start); | ||||||
|  |     if (/[^\w$_-]/.test(word)) { | ||||||
|  |       word = ""; start = end = cur.ch; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var spec = CodeMirror.resolveMode("text/css"); | ||||||
|  |  | ||||||
|  |     var result = []; | ||||||
|  |     function add(keywords) { | ||||||
|  |       for (var name in keywords) | ||||||
|  |         if (!word || name.lastIndexOf(word, 0) == 0) | ||||||
|  |           result.push(name); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var st = inner.state.state; | ||||||
|  |     if (st == "pseudo" || token.type == "variable-3") { | ||||||
|  |       add(pseudoClasses); | ||||||
|  |     } else if (st == "block" || st == "maybeprop") { | ||||||
|  |       add(spec.propertyKeywords); | ||||||
|  |     } else if (st == "prop" || st == "parens" || st == "at" || st == "params") { | ||||||
|  |       add(spec.valueKeywords); | ||||||
|  |       add(spec.colorKeywords); | ||||||
|  |     } else if (st == "media" || st == "media_parens") { | ||||||
|  |       add(spec.mediaTypes); | ||||||
|  |       add(spec.mediaFeatures); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (result.length) return { | ||||||
|  |       list: result, | ||||||
|  |       from: CodeMirror.Pos(cur.line, start), | ||||||
|  |       to: CodeMirror.Pos(cur.line, end) | ||||||
|  |     }; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										350
									
								
								passbook/admin/static/codemirror/addon/hint/html-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										350
									
								
								passbook/admin/static/codemirror/addon/hint/html-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,350 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("./xml-hint")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "./xml-hint"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var langs = "ab aa af ak sq am ar an hy as av ae ay az bm ba eu be bn bh bi bs br bg my ca ch ce ny zh cv kw co cr hr cs da dv nl dz en eo et ee fo fj fi fr ff gl ka de el gn gu ht ha he hz hi ho hu ia id ie ga ig ik io is it iu ja jv kl kn kr ks kk km ki rw ky kv kg ko ku kj la lb lg li ln lo lt lu lv gv mk mg ms ml mt mi mr mh mn na nv nb nd ne ng nn no ii nr oc oj cu om or os pa pi fa pl ps pt qu rm rn ro ru sa sc sd se sm sg sr gd sn si sk sl so st es su sw ss sv ta te tg th ti bo tk tl tn to tr ts tt tw ty ug uk ur uz ve vi vo wa cy wo fy xh yi yo za zu".split(" "); | ||||||
|  |   var targets = ["_blank", "_self", "_top", "_parent"]; | ||||||
|  |   var charsets = ["ascii", "utf-8", "utf-16", "latin1", "latin1"]; | ||||||
|  |   var methods = ["get", "post", "put", "delete"]; | ||||||
|  |   var encs = ["application/x-www-form-urlencoded", "multipart/form-data", "text/plain"]; | ||||||
|  |   var media = ["all", "screen", "print", "embossed", "braille", "handheld", "print", "projection", "screen", "tty", "tv", "speech", | ||||||
|  |                "3d-glasses", "resolution [>][<][=] [X]", "device-aspect-ratio: X/Y", "orientation:portrait", | ||||||
|  |                "orientation:landscape", "device-height: [X]", "device-width: [X]"]; | ||||||
|  |   var s = { attrs: {} }; // Simple tag, reused for a whole lot of tags | ||||||
|  |  | ||||||
|  |   var data = { | ||||||
|  |     a: { | ||||||
|  |       attrs: { | ||||||
|  |         href: null, ping: null, type: null, | ||||||
|  |         media: media, | ||||||
|  |         target: targets, | ||||||
|  |         hreflang: langs | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     abbr: s, | ||||||
|  |     acronym: s, | ||||||
|  |     address: s, | ||||||
|  |     applet: s, | ||||||
|  |     area: { | ||||||
|  |       attrs: { | ||||||
|  |         alt: null, coords: null, href: null, target: null, ping: null, | ||||||
|  |         media: media, hreflang: langs, type: null, | ||||||
|  |         shape: ["default", "rect", "circle", "poly"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     article: s, | ||||||
|  |     aside: s, | ||||||
|  |     audio: { | ||||||
|  |       attrs: { | ||||||
|  |         src: null, mediagroup: null, | ||||||
|  |         crossorigin: ["anonymous", "use-credentials"], | ||||||
|  |         preload: ["none", "metadata", "auto"], | ||||||
|  |         autoplay: ["", "autoplay"], | ||||||
|  |         loop: ["", "loop"], | ||||||
|  |         controls: ["", "controls"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     b: s, | ||||||
|  |     base: { attrs: { href: null, target: targets } }, | ||||||
|  |     basefont: s, | ||||||
|  |     bdi: s, | ||||||
|  |     bdo: s, | ||||||
|  |     big: s, | ||||||
|  |     blockquote: { attrs: { cite: null } }, | ||||||
|  |     body: s, | ||||||
|  |     br: s, | ||||||
|  |     button: { | ||||||
|  |       attrs: { | ||||||
|  |         form: null, formaction: null, name: null, value: null, | ||||||
|  |         autofocus: ["", "autofocus"], | ||||||
|  |         disabled: ["", "autofocus"], | ||||||
|  |         formenctype: encs, | ||||||
|  |         formmethod: methods, | ||||||
|  |         formnovalidate: ["", "novalidate"], | ||||||
|  |         formtarget: targets, | ||||||
|  |         type: ["submit", "reset", "button"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     canvas: { attrs: { width: null, height: null } }, | ||||||
|  |     caption: s, | ||||||
|  |     center: s, | ||||||
|  |     cite: s, | ||||||
|  |     code: s, | ||||||
|  |     col: { attrs: { span: null } }, | ||||||
|  |     colgroup: { attrs: { span: null } }, | ||||||
|  |     command: { | ||||||
|  |       attrs: { | ||||||
|  |         type: ["command", "checkbox", "radio"], | ||||||
|  |         label: null, icon: null, radiogroup: null, command: null, title: null, | ||||||
|  |         disabled: ["", "disabled"], | ||||||
|  |         checked: ["", "checked"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     data: { attrs: { value: null } }, | ||||||
|  |     datagrid: { attrs: { disabled: ["", "disabled"], multiple: ["", "multiple"] } }, | ||||||
|  |     datalist: { attrs: { data: null } }, | ||||||
|  |     dd: s, | ||||||
|  |     del: { attrs: { cite: null, datetime: null } }, | ||||||
|  |     details: { attrs: { open: ["", "open"] } }, | ||||||
|  |     dfn: s, | ||||||
|  |     dir: s, | ||||||
|  |     div: s, | ||||||
|  |     dl: s, | ||||||
|  |     dt: s, | ||||||
|  |     em: s, | ||||||
|  |     embed: { attrs: { src: null, type: null, width: null, height: null } }, | ||||||
|  |     eventsource: { attrs: { src: null } }, | ||||||
|  |     fieldset: { attrs: { disabled: ["", "disabled"], form: null, name: null } }, | ||||||
|  |     figcaption: s, | ||||||
|  |     figure: s, | ||||||
|  |     font: s, | ||||||
|  |     footer: s, | ||||||
|  |     form: { | ||||||
|  |       attrs: { | ||||||
|  |         action: null, name: null, | ||||||
|  |         "accept-charset": charsets, | ||||||
|  |         autocomplete: ["on", "off"], | ||||||
|  |         enctype: encs, | ||||||
|  |         method: methods, | ||||||
|  |         novalidate: ["", "novalidate"], | ||||||
|  |         target: targets | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     frame: s, | ||||||
|  |     frameset: s, | ||||||
|  |     h1: s, h2: s, h3: s, h4: s, h5: s, h6: s, | ||||||
|  |     head: { | ||||||
|  |       attrs: {}, | ||||||
|  |       children: ["title", "base", "link", "style", "meta", "script", "noscript", "command"] | ||||||
|  |     }, | ||||||
|  |     header: s, | ||||||
|  |     hgroup: s, | ||||||
|  |     hr: s, | ||||||
|  |     html: { | ||||||
|  |       attrs: { manifest: null }, | ||||||
|  |       children: ["head", "body"] | ||||||
|  |     }, | ||||||
|  |     i: s, | ||||||
|  |     iframe: { | ||||||
|  |       attrs: { | ||||||
|  |         src: null, srcdoc: null, name: null, width: null, height: null, | ||||||
|  |         sandbox: ["allow-top-navigation", "allow-same-origin", "allow-forms", "allow-scripts"], | ||||||
|  |         seamless: ["", "seamless"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     img: { | ||||||
|  |       attrs: { | ||||||
|  |         alt: null, src: null, ismap: null, usemap: null, width: null, height: null, | ||||||
|  |         crossorigin: ["anonymous", "use-credentials"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     input: { | ||||||
|  |       attrs: { | ||||||
|  |         alt: null, dirname: null, form: null, formaction: null, | ||||||
|  |         height: null, list: null, max: null, maxlength: null, min: null, | ||||||
|  |         name: null, pattern: null, placeholder: null, size: null, src: null, | ||||||
|  |         step: null, value: null, width: null, | ||||||
|  |         accept: ["audio/*", "video/*", "image/*"], | ||||||
|  |         autocomplete: ["on", "off"], | ||||||
|  |         autofocus: ["", "autofocus"], | ||||||
|  |         checked: ["", "checked"], | ||||||
|  |         disabled: ["", "disabled"], | ||||||
|  |         formenctype: encs, | ||||||
|  |         formmethod: methods, | ||||||
|  |         formnovalidate: ["", "novalidate"], | ||||||
|  |         formtarget: targets, | ||||||
|  |         multiple: ["", "multiple"], | ||||||
|  |         readonly: ["", "readonly"], | ||||||
|  |         required: ["", "required"], | ||||||
|  |         type: ["hidden", "text", "search", "tel", "url", "email", "password", "datetime", "date", "month", | ||||||
|  |                "week", "time", "datetime-local", "number", "range", "color", "checkbox", "radio", | ||||||
|  |                "file", "submit", "image", "reset", "button"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     ins: { attrs: { cite: null, datetime: null } }, | ||||||
|  |     kbd: s, | ||||||
|  |     keygen: { | ||||||
|  |       attrs: { | ||||||
|  |         challenge: null, form: null, name: null, | ||||||
|  |         autofocus: ["", "autofocus"], | ||||||
|  |         disabled: ["", "disabled"], | ||||||
|  |         keytype: ["RSA"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     label: { attrs: { "for": null, form: null } }, | ||||||
|  |     legend: s, | ||||||
|  |     li: { attrs: { value: null } }, | ||||||
|  |     link: { | ||||||
|  |       attrs: { | ||||||
|  |         href: null, type: null, | ||||||
|  |         hreflang: langs, | ||||||
|  |         media: media, | ||||||
|  |         sizes: ["all", "16x16", "16x16 32x32", "16x16 32x32 64x64"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     map: { attrs: { name: null } }, | ||||||
|  |     mark: s, | ||||||
|  |     menu: { attrs: { label: null, type: ["list", "context", "toolbar"] } }, | ||||||
|  |     meta: { | ||||||
|  |       attrs: { | ||||||
|  |         content: null, | ||||||
|  |         charset: charsets, | ||||||
|  |         name: ["viewport", "application-name", "author", "description", "generator", "keywords"], | ||||||
|  |         "http-equiv": ["content-language", "content-type", "default-style", "refresh"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     meter: { attrs: { value: null, min: null, low: null, high: null, max: null, optimum: null } }, | ||||||
|  |     nav: s, | ||||||
|  |     noframes: s, | ||||||
|  |     noscript: s, | ||||||
|  |     object: { | ||||||
|  |       attrs: { | ||||||
|  |         data: null, type: null, name: null, usemap: null, form: null, width: null, height: null, | ||||||
|  |         typemustmatch: ["", "typemustmatch"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     ol: { attrs: { reversed: ["", "reversed"], start: null, type: ["1", "a", "A", "i", "I"] } }, | ||||||
|  |     optgroup: { attrs: { disabled: ["", "disabled"], label: null } }, | ||||||
|  |     option: { attrs: { disabled: ["", "disabled"], label: null, selected: ["", "selected"], value: null } }, | ||||||
|  |     output: { attrs: { "for": null, form: null, name: null } }, | ||||||
|  |     p: s, | ||||||
|  |     param: { attrs: { name: null, value: null } }, | ||||||
|  |     pre: s, | ||||||
|  |     progress: { attrs: { value: null, max: null } }, | ||||||
|  |     q: { attrs: { cite: null } }, | ||||||
|  |     rp: s, | ||||||
|  |     rt: s, | ||||||
|  |     ruby: s, | ||||||
|  |     s: s, | ||||||
|  |     samp: s, | ||||||
|  |     script: { | ||||||
|  |       attrs: { | ||||||
|  |         type: ["text/javascript"], | ||||||
|  |         src: null, | ||||||
|  |         async: ["", "async"], | ||||||
|  |         defer: ["", "defer"], | ||||||
|  |         charset: charsets | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     section: s, | ||||||
|  |     select: { | ||||||
|  |       attrs: { | ||||||
|  |         form: null, name: null, size: null, | ||||||
|  |         autofocus: ["", "autofocus"], | ||||||
|  |         disabled: ["", "disabled"], | ||||||
|  |         multiple: ["", "multiple"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     small: s, | ||||||
|  |     source: { attrs: { src: null, type: null, media: null } }, | ||||||
|  |     span: s, | ||||||
|  |     strike: s, | ||||||
|  |     strong: s, | ||||||
|  |     style: { | ||||||
|  |       attrs: { | ||||||
|  |         type: ["text/css"], | ||||||
|  |         media: media, | ||||||
|  |         scoped: null | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     sub: s, | ||||||
|  |     summary: s, | ||||||
|  |     sup: s, | ||||||
|  |     table: s, | ||||||
|  |     tbody: s, | ||||||
|  |     td: { attrs: { colspan: null, rowspan: null, headers: null } }, | ||||||
|  |     textarea: { | ||||||
|  |       attrs: { | ||||||
|  |         dirname: null, form: null, maxlength: null, name: null, placeholder: null, | ||||||
|  |         rows: null, cols: null, | ||||||
|  |         autofocus: ["", "autofocus"], | ||||||
|  |         disabled: ["", "disabled"], | ||||||
|  |         readonly: ["", "readonly"], | ||||||
|  |         required: ["", "required"], | ||||||
|  |         wrap: ["soft", "hard"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     tfoot: s, | ||||||
|  |     th: { attrs: { colspan: null, rowspan: null, headers: null, scope: ["row", "col", "rowgroup", "colgroup"] } }, | ||||||
|  |     thead: s, | ||||||
|  |     time: { attrs: { datetime: null } }, | ||||||
|  |     title: s, | ||||||
|  |     tr: s, | ||||||
|  |     track: { | ||||||
|  |       attrs: { | ||||||
|  |         src: null, label: null, "default": null, | ||||||
|  |         kind: ["subtitles", "captions", "descriptions", "chapters", "metadata"], | ||||||
|  |         srclang: langs | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     tt: s, | ||||||
|  |     u: s, | ||||||
|  |     ul: s, | ||||||
|  |     "var": s, | ||||||
|  |     video: { | ||||||
|  |       attrs: { | ||||||
|  |         src: null, poster: null, width: null, height: null, | ||||||
|  |         crossorigin: ["anonymous", "use-credentials"], | ||||||
|  |         preload: ["auto", "metadata", "none"], | ||||||
|  |         autoplay: ["", "autoplay"], | ||||||
|  |         mediagroup: ["movie"], | ||||||
|  |         muted: ["", "muted"], | ||||||
|  |         controls: ["", "controls"] | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     wbr: s | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   var globalAttrs = { | ||||||
|  |     accesskey: ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", "0", "1", "2", "3", "4", "5", "6", "7", "8", "9"], | ||||||
|  |     "class": null, | ||||||
|  |     contenteditable: ["true", "false"], | ||||||
|  |     contextmenu: null, | ||||||
|  |     dir: ["ltr", "rtl", "auto"], | ||||||
|  |     draggable: ["true", "false", "auto"], | ||||||
|  |     dropzone: ["copy", "move", "link", "string:", "file:"], | ||||||
|  |     hidden: ["hidden"], | ||||||
|  |     id: null, | ||||||
|  |     inert: ["inert"], | ||||||
|  |     itemid: null, | ||||||
|  |     itemprop: null, | ||||||
|  |     itemref: null, | ||||||
|  |     itemscope: ["itemscope"], | ||||||
|  |     itemtype: null, | ||||||
|  |     lang: ["en", "es"], | ||||||
|  |     spellcheck: ["true", "false"], | ||||||
|  |     autocorrect: ["true", "false"], | ||||||
|  |     autocapitalize: ["true", "false"], | ||||||
|  |     style: null, | ||||||
|  |     tabindex: ["1", "2", "3", "4", "5", "6", "7", "8", "9"], | ||||||
|  |     title: null, | ||||||
|  |     translate: ["yes", "no"], | ||||||
|  |     onclick: null, | ||||||
|  |     rel: ["stylesheet", "alternate", "author", "bookmark", "help", "license", "next", "nofollow", "noreferrer", "prefetch", "prev", "search", "tag"] | ||||||
|  |   }; | ||||||
|  |   function populate(obj) { | ||||||
|  |     for (var attr in globalAttrs) if (globalAttrs.hasOwnProperty(attr)) | ||||||
|  |       obj.attrs[attr] = globalAttrs[attr]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   populate(s); | ||||||
|  |   for (var tag in data) if (data.hasOwnProperty(tag) && data[tag] != s) | ||||||
|  |     populate(data[tag]); | ||||||
|  |  | ||||||
|  |   CodeMirror.htmlSchema = data; | ||||||
|  |   function htmlHint(cm, options) { | ||||||
|  |     var local = {schemaInfo: data}; | ||||||
|  |     if (options) for (var opt in options) local[opt] = options[opt]; | ||||||
|  |     return CodeMirror.hint.xml(cm, local); | ||||||
|  |   } | ||||||
|  |   CodeMirror.registerHelper("hint", "html", htmlHint); | ||||||
|  | }); | ||||||
							
								
								
									
										157
									
								
								passbook/admin/static/codemirror/addon/hint/javascript-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								passbook/admin/static/codemirror/addon/hint/javascript-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,157 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   function forEach(arr, f) { | ||||||
|  |     for (var i = 0, e = arr.length; i < e; ++i) f(arr[i]); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function arrayContains(arr, item) { | ||||||
|  |     if (!Array.prototype.indexOf) { | ||||||
|  |       var i = arr.length; | ||||||
|  |       while (i--) { | ||||||
|  |         if (arr[i] === item) { | ||||||
|  |           return true; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |     return arr.indexOf(item) != -1; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function scriptHint(editor, keywords, getToken, options) { | ||||||
|  |     // Find the token at the cursor | ||||||
|  |     var cur = editor.getCursor(), token = getToken(editor, cur); | ||||||
|  |     if (/\b(?:string|comment)\b/.test(token.type)) return; | ||||||
|  |     var innerMode = CodeMirror.innerMode(editor.getMode(), token.state); | ||||||
|  |     if (innerMode.mode.helperType === "json") return; | ||||||
|  |     token.state = innerMode.state; | ||||||
|  |  | ||||||
|  |     // If it's not a 'word-style' token, ignore the token. | ||||||
|  |     if (!/^[\w$_]*$/.test(token.string)) { | ||||||
|  |       token = {start: cur.ch, end: cur.ch, string: "", state: token.state, | ||||||
|  |                type: token.string == "." ? "property" : null}; | ||||||
|  |     } else if (token.end > cur.ch) { | ||||||
|  |       token.end = cur.ch; | ||||||
|  |       token.string = token.string.slice(0, cur.ch - token.start); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var tprop = token; | ||||||
|  |     // If it is a property, find out what it is a property of. | ||||||
|  |     while (tprop.type == "property") { | ||||||
|  |       tprop = getToken(editor, Pos(cur.line, tprop.start)); | ||||||
|  |       if (tprop.string != ".") return; | ||||||
|  |       tprop = getToken(editor, Pos(cur.line, tprop.start)); | ||||||
|  |       if (!context) var context = []; | ||||||
|  |       context.push(tprop); | ||||||
|  |     } | ||||||
|  |     return {list: getCompletions(token, context, keywords, options), | ||||||
|  |             from: Pos(cur.line, token.start), | ||||||
|  |             to: Pos(cur.line, token.end)}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function javascriptHint(editor, options) { | ||||||
|  |     return scriptHint(editor, javascriptKeywords, | ||||||
|  |                       function (e, cur) {return e.getTokenAt(cur);}, | ||||||
|  |                       options); | ||||||
|  |   }; | ||||||
|  |   CodeMirror.registerHelper("hint", "javascript", javascriptHint); | ||||||
|  |  | ||||||
|  |   function getCoffeeScriptToken(editor, cur) { | ||||||
|  |   // This getToken, it is for coffeescript, imitates the behavior of | ||||||
|  |   // getTokenAt method in javascript.js, that is, returning "property" | ||||||
|  |   // type and treat "." as indepenent token. | ||||||
|  |     var token = editor.getTokenAt(cur); | ||||||
|  |     if (cur.ch == token.start + 1 && token.string.charAt(0) == '.') { | ||||||
|  |       token.end = token.start; | ||||||
|  |       token.string = '.'; | ||||||
|  |       token.type = "property"; | ||||||
|  |     } | ||||||
|  |     else if (/^\.[\w$_]*$/.test(token.string)) { | ||||||
|  |       token.type = "property"; | ||||||
|  |       token.start++; | ||||||
|  |       token.string = token.string.replace(/\./, ''); | ||||||
|  |     } | ||||||
|  |     return token; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function coffeescriptHint(editor, options) { | ||||||
|  |     return scriptHint(editor, coffeescriptKeywords, getCoffeeScriptToken, options); | ||||||
|  |   } | ||||||
|  |   CodeMirror.registerHelper("hint", "coffeescript", coffeescriptHint); | ||||||
|  |  | ||||||
|  |   var stringProps = ("charAt charCodeAt indexOf lastIndexOf substring substr slice trim trimLeft trimRight " + | ||||||
|  |                      "toUpperCase toLowerCase split concat match replace search").split(" "); | ||||||
|  |   var arrayProps = ("length concat join splice push pop shift unshift slice reverse sort indexOf " + | ||||||
|  |                     "lastIndexOf every some filter forEach map reduce reduceRight ").split(" "); | ||||||
|  |   var funcProps = "prototype apply call bind".split(" "); | ||||||
|  |   var javascriptKeywords = ("break case catch class const continue debugger default delete do else export extends false finally for function " + | ||||||
|  |                   "if in import instanceof new null return super switch this throw true try typeof var void while with yield").split(" "); | ||||||
|  |   var coffeescriptKeywords = ("and break catch class continue delete do else extends false finally for " + | ||||||
|  |                   "if in instanceof isnt new no not null of off on or return switch then throw true try typeof until void while with yes").split(" "); | ||||||
|  |  | ||||||
|  |   function forAllProps(obj, callback) { | ||||||
|  |     if (!Object.getOwnPropertyNames || !Object.getPrototypeOf) { | ||||||
|  |       for (var name in obj) callback(name) | ||||||
|  |     } else { | ||||||
|  |       for (var o = obj; o; o = Object.getPrototypeOf(o)) | ||||||
|  |         Object.getOwnPropertyNames(o).forEach(callback) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getCompletions(token, context, keywords, options) { | ||||||
|  |     var found = [], start = token.string, global = options && options.globalScope || window; | ||||||
|  |     function maybeAdd(str) { | ||||||
|  |       if (str.lastIndexOf(start, 0) == 0 && !arrayContains(found, str)) found.push(str); | ||||||
|  |     } | ||||||
|  |     function gatherCompletions(obj) { | ||||||
|  |       if (typeof obj == "string") forEach(stringProps, maybeAdd); | ||||||
|  |       else if (obj instanceof Array) forEach(arrayProps, maybeAdd); | ||||||
|  |       else if (obj instanceof Function) forEach(funcProps, maybeAdd); | ||||||
|  |       forAllProps(obj, maybeAdd) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (context && context.length) { | ||||||
|  |       // If this is a property, see if it belongs to some object we can | ||||||
|  |       // find in the current environment. | ||||||
|  |       var obj = context.pop(), base; | ||||||
|  |       if (obj.type && obj.type.indexOf("variable") === 0) { | ||||||
|  |         if (options && options.additionalContext) | ||||||
|  |           base = options.additionalContext[obj.string]; | ||||||
|  |         if (!options || options.useGlobalScope !== false) | ||||||
|  |           base = base || global[obj.string]; | ||||||
|  |       } else if (obj.type == "string") { | ||||||
|  |         base = ""; | ||||||
|  |       } else if (obj.type == "atom") { | ||||||
|  |         base = 1; | ||||||
|  |       } else if (obj.type == "function") { | ||||||
|  |         if (global.jQuery != null && (obj.string == '$' || obj.string == 'jQuery') && | ||||||
|  |             (typeof global.jQuery == 'function')) | ||||||
|  |           base = global.jQuery(); | ||||||
|  |         else if (global._ != null && (obj.string == '_') && (typeof global._ == 'function')) | ||||||
|  |           base = global._(); | ||||||
|  |       } | ||||||
|  |       while (base != null && context.length) | ||||||
|  |         base = base[context.pop().string]; | ||||||
|  |       if (base != null) gatherCompletions(base); | ||||||
|  |     } else { | ||||||
|  |       // If not, just look in the global object and any local scope | ||||||
|  |       // (reading into JS mode internals to get at the local and global variables) | ||||||
|  |       for (var v = token.state.localVars; v; v = v.next) maybeAdd(v.name); | ||||||
|  |       for (var v = token.state.globalVars; v; v = v.next) maybeAdd(v.name); | ||||||
|  |       if (!options || options.useGlobalScope !== false) | ||||||
|  |         gatherCompletions(global); | ||||||
|  |       forEach(keywords, maybeAdd); | ||||||
|  |     } | ||||||
|  |     return found; | ||||||
|  |   } | ||||||
|  | }); | ||||||
							
								
								
									
										36
									
								
								passbook/admin/static/codemirror/addon/hint/show-hint.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								passbook/admin/static/codemirror/addon/hint/show-hint.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | .CodeMirror-hints { | ||||||
|  |   position: absolute; | ||||||
|  |   z-index: 10; | ||||||
|  |   overflow: hidden; | ||||||
|  |   list-style: none; | ||||||
|  |  | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 2px; | ||||||
|  |  | ||||||
|  |   -webkit-box-shadow: 2px 3px 5px rgba(0,0,0,.2); | ||||||
|  |   -moz-box-shadow: 2px 3px 5px rgba(0,0,0,.2); | ||||||
|  |   box-shadow: 2px 3px 5px rgba(0,0,0,.2); | ||||||
|  |   border-radius: 3px; | ||||||
|  |   border: 1px solid silver; | ||||||
|  |  | ||||||
|  |   background: white; | ||||||
|  |   font-size: 90%; | ||||||
|  |   font-family: monospace; | ||||||
|  |  | ||||||
|  |   max-height: 20em; | ||||||
|  |   overflow-y: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .CodeMirror-hint { | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 0 4px; | ||||||
|  |   border-radius: 2px; | ||||||
|  |   white-space: pre; | ||||||
|  |   color: black; | ||||||
|  |   cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | li.CodeMirror-hint-active { | ||||||
|  |   background: #08f; | ||||||
|  |   color: white; | ||||||
|  | } | ||||||
							
								
								
									
										460
									
								
								passbook/admin/static/codemirror/addon/hint/show-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										460
									
								
								passbook/admin/static/codemirror/addon/hint/show-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,460 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var HINT_ELEMENT_CLASS        = "CodeMirror-hint"; | ||||||
|  |   var ACTIVE_HINT_ELEMENT_CLASS = "CodeMirror-hint-active"; | ||||||
|  |  | ||||||
|  |   // This is the old interface, kept around for now to stay | ||||||
|  |   // backwards-compatible. | ||||||
|  |   CodeMirror.showHint = function(cm, getHints, options) { | ||||||
|  |     if (!getHints) return cm.showHint(options); | ||||||
|  |     if (options && options.async) getHints.async = true; | ||||||
|  |     var newOpts = {hint: getHints}; | ||||||
|  |     if (options) for (var prop in options) newOpts[prop] = options[prop]; | ||||||
|  |     return cm.showHint(newOpts); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("showHint", function(options) { | ||||||
|  |     options = parseOptions(this, this.getCursor("start"), options); | ||||||
|  |     var selections = this.listSelections() | ||||||
|  |     if (selections.length > 1) return; | ||||||
|  |     // By default, don't allow completion when something is selected. | ||||||
|  |     // A hint function can have a `supportsSelection` property to | ||||||
|  |     // indicate that it can handle selections. | ||||||
|  |     if (this.somethingSelected()) { | ||||||
|  |       if (!options.hint.supportsSelection) return; | ||||||
|  |       // Don't try with cross-line selections | ||||||
|  |       for (var i = 0; i < selections.length; i++) | ||||||
|  |         if (selections[i].head.line != selections[i].anchor.line) return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (this.state.completionActive) this.state.completionActive.close(); | ||||||
|  |     var completion = this.state.completionActive = new Completion(this, options); | ||||||
|  |     if (!completion.options.hint) return; | ||||||
|  |  | ||||||
|  |     CodeMirror.signal(this, "startCompletion", this); | ||||||
|  |     completion.update(true); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.defineExtension("closeHint", function() { | ||||||
|  |     if (this.state.completionActive) this.state.completionActive.close() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   function Completion(cm, options) { | ||||||
|  |     this.cm = cm; | ||||||
|  |     this.options = options; | ||||||
|  |     this.widget = null; | ||||||
|  |     this.debounce = 0; | ||||||
|  |     this.tick = 0; | ||||||
|  |     this.startPos = this.cm.getCursor("start"); | ||||||
|  |     this.startLen = this.cm.getLine(this.startPos.line).length - this.cm.getSelection().length; | ||||||
|  |  | ||||||
|  |     var self = this; | ||||||
|  |     cm.on("cursorActivity", this.activityFunc = function() { self.cursorActivity(); }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   var requestAnimationFrame = window.requestAnimationFrame || function(fn) { | ||||||
|  |     return setTimeout(fn, 1000/60); | ||||||
|  |   }; | ||||||
|  |   var cancelAnimationFrame = window.cancelAnimationFrame || clearTimeout; | ||||||
|  |  | ||||||
|  |   Completion.prototype = { | ||||||
|  |     close: function() { | ||||||
|  |       if (!this.active()) return; | ||||||
|  |       this.cm.state.completionActive = null; | ||||||
|  |       this.tick = null; | ||||||
|  |       this.cm.off("cursorActivity", this.activityFunc); | ||||||
|  |  | ||||||
|  |       if (this.widget && this.data) CodeMirror.signal(this.data, "close"); | ||||||
|  |       if (this.widget) this.widget.close(); | ||||||
|  |       CodeMirror.signal(this.cm, "endCompletion", this.cm); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     active: function() { | ||||||
|  |       return this.cm.state.completionActive == this; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     pick: function(data, i) { | ||||||
|  |       var completion = data.list[i]; | ||||||
|  |       if (completion.hint) completion.hint(this.cm, data, completion); | ||||||
|  |       else this.cm.replaceRange(getText(completion), completion.from || data.from, | ||||||
|  |                                 completion.to || data.to, "complete"); | ||||||
|  |       CodeMirror.signal(data, "pick", completion); | ||||||
|  |       this.close(); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     cursorActivity: function() { | ||||||
|  |       if (this.debounce) { | ||||||
|  |         cancelAnimationFrame(this.debounce); | ||||||
|  |         this.debounce = 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       var pos = this.cm.getCursor(), line = this.cm.getLine(pos.line); | ||||||
|  |       if (pos.line != this.startPos.line || line.length - pos.ch != this.startLen - this.startPos.ch || | ||||||
|  |           pos.ch < this.startPos.ch || this.cm.somethingSelected() || | ||||||
|  |           (!pos.ch || this.options.closeCharacters.test(line.charAt(pos.ch - 1)))) { | ||||||
|  |         this.close(); | ||||||
|  |       } else { | ||||||
|  |         var self = this; | ||||||
|  |         this.debounce = requestAnimationFrame(function() {self.update();}); | ||||||
|  |         if (this.widget) this.widget.disable(); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     update: function(first) { | ||||||
|  |       if (this.tick == null) return | ||||||
|  |       var self = this, myTick = ++this.tick | ||||||
|  |       fetchHints(this.options.hint, this.cm, this.options, function(data) { | ||||||
|  |         if (self.tick == myTick) self.finishUpdate(data, first) | ||||||
|  |       }) | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     finishUpdate: function(data, first) { | ||||||
|  |       if (this.data) CodeMirror.signal(this.data, "update"); | ||||||
|  |  | ||||||
|  |       var picked = (this.widget && this.widget.picked) || (first && this.options.completeSingle); | ||||||
|  |       if (this.widget) this.widget.close(); | ||||||
|  |  | ||||||
|  |       this.data = data; | ||||||
|  |  | ||||||
|  |       if (data && data.list.length) { | ||||||
|  |         if (picked && data.list.length == 1) { | ||||||
|  |           this.pick(data, 0); | ||||||
|  |         } else { | ||||||
|  |           this.widget = new Widget(this, data); | ||||||
|  |           CodeMirror.signal(data, "shown"); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   function parseOptions(cm, pos, options) { | ||||||
|  |     var editor = cm.options.hintOptions; | ||||||
|  |     var out = {}; | ||||||
|  |     for (var prop in defaultOptions) out[prop] = defaultOptions[prop]; | ||||||
|  |     if (editor) for (var prop in editor) | ||||||
|  |       if (editor[prop] !== undefined) out[prop] = editor[prop]; | ||||||
|  |     if (options) for (var prop in options) | ||||||
|  |       if (options[prop] !== undefined) out[prop] = options[prop]; | ||||||
|  |     if (out.hint.resolve) out.hint = out.hint.resolve(cm, pos) | ||||||
|  |     return out; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getText(completion) { | ||||||
|  |     if (typeof completion == "string") return completion; | ||||||
|  |     else return completion.text; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function buildKeyMap(completion, handle) { | ||||||
|  |     var baseMap = { | ||||||
|  |       Up: function() {handle.moveFocus(-1);}, | ||||||
|  |       Down: function() {handle.moveFocus(1);}, | ||||||
|  |       PageUp: function() {handle.moveFocus(-handle.menuSize() + 1, true);}, | ||||||
|  |       PageDown: function() {handle.moveFocus(handle.menuSize() - 1, true);}, | ||||||
|  |       Home: function() {handle.setFocus(0);}, | ||||||
|  |       End: function() {handle.setFocus(handle.length - 1);}, | ||||||
|  |       Enter: handle.pick, | ||||||
|  |       Tab: handle.pick, | ||||||
|  |       Esc: handle.close | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     var mac = /Mac/.test(navigator.platform); | ||||||
|  |  | ||||||
|  |     if (mac) { | ||||||
|  |       baseMap["Ctrl-P"] = function() {handle.moveFocus(-1);}; | ||||||
|  |       baseMap["Ctrl-N"] = function() {handle.moveFocus(1);}; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var custom = completion.options.customKeys; | ||||||
|  |     var ourMap = custom ? {} : baseMap; | ||||||
|  |     function addBinding(key, val) { | ||||||
|  |       var bound; | ||||||
|  |       if (typeof val != "string") | ||||||
|  |         bound = function(cm) { return val(cm, handle); }; | ||||||
|  |       // This mechanism is deprecated | ||||||
|  |       else if (baseMap.hasOwnProperty(val)) | ||||||
|  |         bound = baseMap[val]; | ||||||
|  |       else | ||||||
|  |         bound = val; | ||||||
|  |       ourMap[key] = bound; | ||||||
|  |     } | ||||||
|  |     if (custom) | ||||||
|  |       for (var key in custom) if (custom.hasOwnProperty(key)) | ||||||
|  |         addBinding(key, custom[key]); | ||||||
|  |     var extra = completion.options.extraKeys; | ||||||
|  |     if (extra) | ||||||
|  |       for (var key in extra) if (extra.hasOwnProperty(key)) | ||||||
|  |         addBinding(key, extra[key]); | ||||||
|  |     return ourMap; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getHintElement(hintsElement, el) { | ||||||
|  |     while (el && el != hintsElement) { | ||||||
|  |       if (el.nodeName.toUpperCase() === "LI" && el.parentNode == hintsElement) return el; | ||||||
|  |       el = el.parentNode; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function Widget(completion, data) { | ||||||
|  |     this.completion = completion; | ||||||
|  |     this.data = data; | ||||||
|  |     this.picked = false; | ||||||
|  |     var widget = this, cm = completion.cm; | ||||||
|  |     var ownerDocument = cm.getInputField().ownerDocument; | ||||||
|  |     var parentWindow = ownerDocument.defaultView || ownerDocument.parentWindow; | ||||||
|  |  | ||||||
|  |     var hints = this.hints = ownerDocument.createElement("ul"); | ||||||
|  |     var theme = completion.cm.options.theme; | ||||||
|  |     hints.className = "CodeMirror-hints " + theme; | ||||||
|  |     this.selectedHint = data.selectedHint || 0; | ||||||
|  |  | ||||||
|  |     var completions = data.list; | ||||||
|  |     for (var i = 0; i < completions.length; ++i) { | ||||||
|  |       var elt = hints.appendChild(ownerDocument.createElement("li")), cur = completions[i]; | ||||||
|  |       var className = HINT_ELEMENT_CLASS + (i != this.selectedHint ? "" : " " + ACTIVE_HINT_ELEMENT_CLASS); | ||||||
|  |       if (cur.className != null) className = cur.className + " " + className; | ||||||
|  |       elt.className = className; | ||||||
|  |       if (cur.render) cur.render(elt, data, cur); | ||||||
|  |       else elt.appendChild(ownerDocument.createTextNode(cur.displayText || getText(cur))); | ||||||
|  |       elt.hintId = i; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var container = completion.options.container || ownerDocument.body; | ||||||
|  |     var pos = cm.cursorCoords(completion.options.alignWithWord ? data.from : null); | ||||||
|  |     var left = pos.left, top = pos.bottom, below = true; | ||||||
|  |     var offsetLeft = 0, offsetTop = 0; | ||||||
|  |     if (container !== ownerDocument.body) { | ||||||
|  |       // We offset the cursor position because left and top are relative to the offsetParent's top left corner. | ||||||
|  |       var isContainerPositioned = ['absolute', 'relative', 'fixed'].indexOf(parentWindow.getComputedStyle(container).position) !== -1; | ||||||
|  |       var offsetParent = isContainerPositioned ? container : container.offsetParent; | ||||||
|  |       var offsetParentPosition = offsetParent.getBoundingClientRect(); | ||||||
|  |       var bodyPosition = ownerDocument.body.getBoundingClientRect(); | ||||||
|  |       offsetLeft = (offsetParentPosition.left - bodyPosition.left - offsetParent.scrollLeft); | ||||||
|  |       offsetTop = (offsetParentPosition.top - bodyPosition.top - offsetParent.scrollTop); | ||||||
|  |     } | ||||||
|  |     hints.style.left = (left - offsetLeft) + "px"; | ||||||
|  |     hints.style.top = (top - offsetTop) + "px"; | ||||||
|  |  | ||||||
|  |     // If we're at the edge of the screen, then we want the menu to appear on the left of the cursor. | ||||||
|  |     var winW = parentWindow.innerWidth || Math.max(ownerDocument.body.offsetWidth, ownerDocument.documentElement.offsetWidth); | ||||||
|  |     var winH = parentWindow.innerHeight || Math.max(ownerDocument.body.offsetHeight, ownerDocument.documentElement.offsetHeight); | ||||||
|  |     container.appendChild(hints); | ||||||
|  |     var box = hints.getBoundingClientRect(), overlapY = box.bottom - winH; | ||||||
|  |     var scrolls = hints.scrollHeight > hints.clientHeight + 1 | ||||||
|  |     var startScroll = cm.getScrollInfo(); | ||||||
|  |  | ||||||
|  |     if (overlapY > 0) { | ||||||
|  |       var height = box.bottom - box.top, curTop = pos.top - (pos.bottom - box.top); | ||||||
|  |       if (curTop - height > 0) { // Fits above cursor | ||||||
|  |         hints.style.top = (top = pos.top - height - offsetTop) + "px"; | ||||||
|  |         below = false; | ||||||
|  |       } else if (height > winH) { | ||||||
|  |         hints.style.height = (winH - 5) + "px"; | ||||||
|  |         hints.style.top = (top = pos.bottom - box.top - offsetTop) + "px"; | ||||||
|  |         var cursor = cm.getCursor(); | ||||||
|  |         if (data.from.ch != cursor.ch) { | ||||||
|  |           pos = cm.cursorCoords(cursor); | ||||||
|  |           hints.style.left = (left = pos.left - offsetLeft) + "px"; | ||||||
|  |           box = hints.getBoundingClientRect(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     var overlapX = box.right - winW; | ||||||
|  |     if (overlapX > 0) { | ||||||
|  |       if (box.right - box.left > winW) { | ||||||
|  |         hints.style.width = (winW - 5) + "px"; | ||||||
|  |         overlapX -= (box.right - box.left) - winW; | ||||||
|  |       } | ||||||
|  |       hints.style.left = (left = pos.left - overlapX - offsetLeft) + "px"; | ||||||
|  |     } | ||||||
|  |     if (scrolls) for (var node = hints.firstChild; node; node = node.nextSibling) | ||||||
|  |       node.style.paddingRight = cm.display.nativeBarWidth + "px" | ||||||
|  |  | ||||||
|  |     cm.addKeyMap(this.keyMap = buildKeyMap(completion, { | ||||||
|  |       moveFocus: function(n, avoidWrap) { widget.changeActive(widget.selectedHint + n, avoidWrap); }, | ||||||
|  |       setFocus: function(n) { widget.changeActive(n); }, | ||||||
|  |       menuSize: function() { return widget.screenAmount(); }, | ||||||
|  |       length: completions.length, | ||||||
|  |       close: function() { completion.close(); }, | ||||||
|  |       pick: function() { widget.pick(); }, | ||||||
|  |       data: data | ||||||
|  |     })); | ||||||
|  |  | ||||||
|  |     if (completion.options.closeOnUnfocus) { | ||||||
|  |       var closingOnBlur; | ||||||
|  |       cm.on("blur", this.onBlur = function() { closingOnBlur = setTimeout(function() { completion.close(); }, 100); }); | ||||||
|  |       cm.on("focus", this.onFocus = function() { clearTimeout(closingOnBlur); }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     cm.on("scroll", this.onScroll = function() { | ||||||
|  |       var curScroll = cm.getScrollInfo(), editor = cm.getWrapperElement().getBoundingClientRect(); | ||||||
|  |       var newTop = top + startScroll.top - curScroll.top; | ||||||
|  |       var point = newTop - (parentWindow.pageYOffset || (ownerDocument.documentElement || ownerDocument.body).scrollTop); | ||||||
|  |       if (!below) point += hints.offsetHeight; | ||||||
|  |       if (point <= editor.top || point >= editor.bottom) return completion.close(); | ||||||
|  |       hints.style.top = newTop + "px"; | ||||||
|  |       hints.style.left = (left + startScroll.left - curScroll.left) + "px"; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     CodeMirror.on(hints, "dblclick", function(e) { | ||||||
|  |       var t = getHintElement(hints, e.target || e.srcElement); | ||||||
|  |       if (t && t.hintId != null) {widget.changeActive(t.hintId); widget.pick();} | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     CodeMirror.on(hints, "click", function(e) { | ||||||
|  |       var t = getHintElement(hints, e.target || e.srcElement); | ||||||
|  |       if (t && t.hintId != null) { | ||||||
|  |         widget.changeActive(t.hintId); | ||||||
|  |         if (completion.options.completeOnSingleClick) widget.pick(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     CodeMirror.on(hints, "mousedown", function() { | ||||||
|  |       setTimeout(function(){cm.focus();}, 20); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     CodeMirror.signal(data, "select", completions[this.selectedHint], hints.childNodes[this.selectedHint]); | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Widget.prototype = { | ||||||
|  |     close: function() { | ||||||
|  |       if (this.completion.widget != this) return; | ||||||
|  |       this.completion.widget = null; | ||||||
|  |       this.hints.parentNode.removeChild(this.hints); | ||||||
|  |       this.completion.cm.removeKeyMap(this.keyMap); | ||||||
|  |  | ||||||
|  |       var cm = this.completion.cm; | ||||||
|  |       if (this.completion.options.closeOnUnfocus) { | ||||||
|  |         cm.off("blur", this.onBlur); | ||||||
|  |         cm.off("focus", this.onFocus); | ||||||
|  |       } | ||||||
|  |       cm.off("scroll", this.onScroll); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     disable: function() { | ||||||
|  |       this.completion.cm.removeKeyMap(this.keyMap); | ||||||
|  |       var widget = this; | ||||||
|  |       this.keyMap = {Enter: function() { widget.picked = true; }}; | ||||||
|  |       this.completion.cm.addKeyMap(this.keyMap); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     pick: function() { | ||||||
|  |       this.completion.pick(this.data, this.selectedHint); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     changeActive: function(i, avoidWrap) { | ||||||
|  |       if (i >= this.data.list.length) | ||||||
|  |         i = avoidWrap ? this.data.list.length - 1 : 0; | ||||||
|  |       else if (i < 0) | ||||||
|  |         i = avoidWrap ? 0  : this.data.list.length - 1; | ||||||
|  |       if (this.selectedHint == i) return; | ||||||
|  |       var node = this.hints.childNodes[this.selectedHint]; | ||||||
|  |       if (node) node.className = node.className.replace(" " + ACTIVE_HINT_ELEMENT_CLASS, ""); | ||||||
|  |       node = this.hints.childNodes[this.selectedHint = i]; | ||||||
|  |       node.className += " " + ACTIVE_HINT_ELEMENT_CLASS; | ||||||
|  |       if (node.offsetTop < this.hints.scrollTop) | ||||||
|  |         this.hints.scrollTop = node.offsetTop - 3; | ||||||
|  |       else if (node.offsetTop + node.offsetHeight > this.hints.scrollTop + this.hints.clientHeight) | ||||||
|  |         this.hints.scrollTop = node.offsetTop + node.offsetHeight - this.hints.clientHeight + 3; | ||||||
|  |       CodeMirror.signal(this.data, "select", this.data.list[this.selectedHint], node); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     screenAmount: function() { | ||||||
|  |       return Math.floor(this.hints.clientHeight / this.hints.firstChild.offsetHeight) || 1; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   function applicableHelpers(cm, helpers) { | ||||||
|  |     if (!cm.somethingSelected()) return helpers | ||||||
|  |     var result = [] | ||||||
|  |     for (var i = 0; i < helpers.length; i++) | ||||||
|  |       if (helpers[i].supportsSelection) result.push(helpers[i]) | ||||||
|  |     return result | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function fetchHints(hint, cm, options, callback) { | ||||||
|  |     if (hint.async) { | ||||||
|  |       hint(cm, callback, options) | ||||||
|  |     } else { | ||||||
|  |       var result = hint(cm, options) | ||||||
|  |       if (result && result.then) result.then(callback) | ||||||
|  |       else callback(result) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function resolveAutoHints(cm, pos) { | ||||||
|  |     var helpers = cm.getHelpers(pos, "hint"), words | ||||||
|  |     if (helpers.length) { | ||||||
|  |       var resolved = function(cm, callback, options) { | ||||||
|  |         var app = applicableHelpers(cm, helpers); | ||||||
|  |         function run(i) { | ||||||
|  |           if (i == app.length) return callback(null) | ||||||
|  |           fetchHints(app[i], cm, options, function(result) { | ||||||
|  |             if (result && result.list.length > 0) callback(result) | ||||||
|  |             else run(i + 1) | ||||||
|  |           }) | ||||||
|  |         } | ||||||
|  |         run(0) | ||||||
|  |       } | ||||||
|  |       resolved.async = true | ||||||
|  |       resolved.supportsSelection = true | ||||||
|  |       return resolved | ||||||
|  |     } else if (words = cm.getHelper(cm.getCursor(), "hintWords")) { | ||||||
|  |       return function(cm) { return CodeMirror.hint.fromList(cm, {words: words}) } | ||||||
|  |     } else if (CodeMirror.hint.anyword) { | ||||||
|  |       return function(cm, options) { return CodeMirror.hint.anyword(cm, options) } | ||||||
|  |     } else { | ||||||
|  |       return function() {} | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "auto", { | ||||||
|  |     resolve: resolveAutoHints | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "fromList", function(cm, options) { | ||||||
|  |     var cur = cm.getCursor(), token = cm.getTokenAt(cur) | ||||||
|  |     var term, from = CodeMirror.Pos(cur.line, token.start), to = cur | ||||||
|  |     if (token.start < cur.ch && /\w/.test(token.string.charAt(cur.ch - token.start - 1))) { | ||||||
|  |       term = token.string.substr(0, cur.ch - token.start) | ||||||
|  |     } else { | ||||||
|  |       term = "" | ||||||
|  |       from = cur | ||||||
|  |     } | ||||||
|  |     var found = []; | ||||||
|  |     for (var i = 0; i < options.words.length; i++) { | ||||||
|  |       var word = options.words[i]; | ||||||
|  |       if (word.slice(0, term.length) == term) | ||||||
|  |         found.push(word); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (found.length) return {list: found, from: from, to: to}; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   CodeMirror.commands.autocomplete = CodeMirror.showHint; | ||||||
|  |  | ||||||
|  |   var defaultOptions = { | ||||||
|  |     hint: CodeMirror.hint.auto, | ||||||
|  |     completeSingle: true, | ||||||
|  |     alignWithWord: true, | ||||||
|  |     closeCharacters: /[\s()\[\]{};:>,]/, | ||||||
|  |     closeOnUnfocus: true, | ||||||
|  |     completeOnSingleClick: true, | ||||||
|  |     container: null, | ||||||
|  |     customKeys: null, | ||||||
|  |     extraKeys: null | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.defineOption("hintOptions", null); | ||||||
|  | }); | ||||||
							
								
								
									
										304
									
								
								passbook/admin/static/codemirror/addon/hint/sql-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										304
									
								
								passbook/admin/static/codemirror/addon/hint/sql-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,304 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("../../mode/sql/sql")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "../../mode/sql/sql"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var tables; | ||||||
|  |   var defaultTable; | ||||||
|  |   var keywords; | ||||||
|  |   var identifierQuote; | ||||||
|  |   var CONS = { | ||||||
|  |     QUERY_DIV: ";", | ||||||
|  |     ALIAS_KEYWORD: "AS" | ||||||
|  |   }; | ||||||
|  |   var Pos = CodeMirror.Pos, cmpPos = CodeMirror.cmpPos; | ||||||
|  |  | ||||||
|  |   function isArray(val) { return Object.prototype.toString.call(val) == "[object Array]" } | ||||||
|  |  | ||||||
|  |   function getKeywords(editor) { | ||||||
|  |     var mode = editor.doc.modeOption; | ||||||
|  |     if (mode === "sql") mode = "text/x-sql"; | ||||||
|  |     return CodeMirror.resolveMode(mode).keywords; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getIdentifierQuote(editor) { | ||||||
|  |     var mode = editor.doc.modeOption; | ||||||
|  |     if (mode === "sql") mode = "text/x-sql"; | ||||||
|  |     return CodeMirror.resolveMode(mode).identifierQuote || "`"; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getText(item) { | ||||||
|  |     return typeof item == "string" ? item : item.text; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function wrapTable(name, value) { | ||||||
|  |     if (isArray(value)) value = {columns: value} | ||||||
|  |     if (!value.text) value.text = name | ||||||
|  |     return value | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function parseTables(input) { | ||||||
|  |     var result = {} | ||||||
|  |     if (isArray(input)) { | ||||||
|  |       for (var i = input.length - 1; i >= 0; i--) { | ||||||
|  |         var item = input[i] | ||||||
|  |         result[getText(item).toUpperCase()] = wrapTable(getText(item), item) | ||||||
|  |       } | ||||||
|  |     } else if (input) { | ||||||
|  |       for (var name in input) | ||||||
|  |         result[name.toUpperCase()] = wrapTable(name, input[name]) | ||||||
|  |     } | ||||||
|  |     return result | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getTable(name) { | ||||||
|  |     return tables[name.toUpperCase()] | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function shallowClone(object) { | ||||||
|  |     var result = {}; | ||||||
|  |     for (var key in object) if (object.hasOwnProperty(key)) | ||||||
|  |       result[key] = object[key]; | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function match(string, word) { | ||||||
|  |     var len = string.length; | ||||||
|  |     var sub = getText(word).substr(0, len); | ||||||
|  |     return string.toUpperCase() === sub.toUpperCase(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function addMatches(result, search, wordlist, formatter) { | ||||||
|  |     if (isArray(wordlist)) { | ||||||
|  |       for (var i = 0; i < wordlist.length; i++) | ||||||
|  |         if (match(search, wordlist[i])) result.push(formatter(wordlist[i])) | ||||||
|  |     } else { | ||||||
|  |       for (var word in wordlist) if (wordlist.hasOwnProperty(word)) { | ||||||
|  |         var val = wordlist[word] | ||||||
|  |         if (!val || val === true) | ||||||
|  |           val = word | ||||||
|  |         else | ||||||
|  |           val = val.displayText ? {text: val.text, displayText: val.displayText} : val.text | ||||||
|  |         if (match(search, val)) result.push(formatter(val)) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function cleanName(name) { | ||||||
|  |     // Get rid name from identifierQuote and preceding dot(.) | ||||||
|  |     if (name.charAt(0) == ".") { | ||||||
|  |       name = name.substr(1); | ||||||
|  |     } | ||||||
|  |     // replace doublicated identifierQuotes with single identifierQuotes | ||||||
|  |     // and remove single identifierQuotes | ||||||
|  |     var nameParts = name.split(identifierQuote+identifierQuote); | ||||||
|  |     for (var i = 0; i < nameParts.length; i++) | ||||||
|  |       nameParts[i] = nameParts[i].replace(new RegExp(identifierQuote,"g"), ""); | ||||||
|  |     return nameParts.join(identifierQuote); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function insertIdentifierQuotes(name) { | ||||||
|  |     var nameParts = getText(name).split("."); | ||||||
|  |     for (var i = 0; i < nameParts.length; i++) | ||||||
|  |       nameParts[i] = identifierQuote + | ||||||
|  |         // doublicate identifierQuotes | ||||||
|  |         nameParts[i].replace(new RegExp(identifierQuote,"g"), identifierQuote+identifierQuote) + | ||||||
|  |         identifierQuote; | ||||||
|  |     var escaped = nameParts.join("."); | ||||||
|  |     if (typeof name == "string") return escaped; | ||||||
|  |     name = shallowClone(name); | ||||||
|  |     name.text = escaped; | ||||||
|  |     return name; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function nameCompletion(cur, token, result, editor) { | ||||||
|  |     // Try to complete table, column names and return start position of completion | ||||||
|  |     var useIdentifierQuotes = false; | ||||||
|  |     var nameParts = []; | ||||||
|  |     var start = token.start; | ||||||
|  |     var cont = true; | ||||||
|  |     while (cont) { | ||||||
|  |       cont = (token.string.charAt(0) == "."); | ||||||
|  |       useIdentifierQuotes = useIdentifierQuotes || (token.string.charAt(0) == identifierQuote); | ||||||
|  |  | ||||||
|  |       start = token.start; | ||||||
|  |       nameParts.unshift(cleanName(token.string)); | ||||||
|  |  | ||||||
|  |       token = editor.getTokenAt(Pos(cur.line, token.start)); | ||||||
|  |       if (token.string == ".") { | ||||||
|  |         cont = true; | ||||||
|  |         token = editor.getTokenAt(Pos(cur.line, token.start)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Try to complete table names | ||||||
|  |     var string = nameParts.join("."); | ||||||
|  |     addMatches(result, string, tables, function(w) { | ||||||
|  |       return useIdentifierQuotes ? insertIdentifierQuotes(w) : w; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     // Try to complete columns from defaultTable | ||||||
|  |     addMatches(result, string, defaultTable, function(w) { | ||||||
|  |       return useIdentifierQuotes ? insertIdentifierQuotes(w) : w; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     // Try to complete columns | ||||||
|  |     string = nameParts.pop(); | ||||||
|  |     var table = nameParts.join("."); | ||||||
|  |  | ||||||
|  |     var alias = false; | ||||||
|  |     var aliasTable = table; | ||||||
|  |     // Check if table is available. If not, find table by Alias | ||||||
|  |     if (!getTable(table)) { | ||||||
|  |       var oldTable = table; | ||||||
|  |       table = findTableByAlias(table, editor); | ||||||
|  |       if (table !== oldTable) alias = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var columns = getTable(table); | ||||||
|  |     if (columns && columns.columns) | ||||||
|  |       columns = columns.columns; | ||||||
|  |  | ||||||
|  |     if (columns) { | ||||||
|  |       addMatches(result, string, columns, function(w) { | ||||||
|  |         var tableInsert = table; | ||||||
|  |         if (alias == true) tableInsert = aliasTable; | ||||||
|  |         if (typeof w == "string") { | ||||||
|  |           w = tableInsert + "." + w; | ||||||
|  |         } else { | ||||||
|  |           w = shallowClone(w); | ||||||
|  |           w.text = tableInsert + "." + w.text; | ||||||
|  |         } | ||||||
|  |         return useIdentifierQuotes ? insertIdentifierQuotes(w) : w; | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return start; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function eachWord(lineText, f) { | ||||||
|  |     var words = lineText.split(/\s+/) | ||||||
|  |     for (var i = 0; i < words.length; i++) | ||||||
|  |       if (words[i]) f(words[i].replace(/[,;]/g, '')) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function findTableByAlias(alias, editor) { | ||||||
|  |     var doc = editor.doc; | ||||||
|  |     var fullQuery = doc.getValue(); | ||||||
|  |     var aliasUpperCase = alias.toUpperCase(); | ||||||
|  |     var previousWord = ""; | ||||||
|  |     var table = ""; | ||||||
|  |     var separator = []; | ||||||
|  |     var validRange = { | ||||||
|  |       start: Pos(0, 0), | ||||||
|  |       end: Pos(editor.lastLine(), editor.getLineHandle(editor.lastLine()).length) | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     //add separator | ||||||
|  |     var indexOfSeparator = fullQuery.indexOf(CONS.QUERY_DIV); | ||||||
|  |     while(indexOfSeparator != -1) { | ||||||
|  |       separator.push(doc.posFromIndex(indexOfSeparator)); | ||||||
|  |       indexOfSeparator = fullQuery.indexOf(CONS.QUERY_DIV, indexOfSeparator+1); | ||||||
|  |     } | ||||||
|  |     separator.unshift(Pos(0, 0)); | ||||||
|  |     separator.push(Pos(editor.lastLine(), editor.getLineHandle(editor.lastLine()).text.length)); | ||||||
|  |  | ||||||
|  |     //find valid range | ||||||
|  |     var prevItem = null; | ||||||
|  |     var current = editor.getCursor() | ||||||
|  |     for (var i = 0; i < separator.length; i++) { | ||||||
|  |       if ((prevItem == null || cmpPos(current, prevItem) > 0) && cmpPos(current, separator[i]) <= 0) { | ||||||
|  |         validRange = {start: prevItem, end: separator[i]}; | ||||||
|  |         break; | ||||||
|  |       } | ||||||
|  |       prevItem = separator[i]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (validRange.start) { | ||||||
|  |       var query = doc.getRange(validRange.start, validRange.end, false); | ||||||
|  |  | ||||||
|  |       for (var i = 0; i < query.length; i++) { | ||||||
|  |         var lineText = query[i]; | ||||||
|  |         eachWord(lineText, function(word) { | ||||||
|  |           var wordUpperCase = word.toUpperCase(); | ||||||
|  |           if (wordUpperCase === aliasUpperCase && getTable(previousWord)) | ||||||
|  |             table = previousWord; | ||||||
|  |           if (wordUpperCase !== CONS.ALIAS_KEYWORD) | ||||||
|  |             previousWord = word; | ||||||
|  |         }); | ||||||
|  |         if (table) break; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return table; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "sql", function(editor, options) { | ||||||
|  |     tables = parseTables(options && options.tables) | ||||||
|  |     var defaultTableName = options && options.defaultTable; | ||||||
|  |     var disableKeywords = options && options.disableKeywords; | ||||||
|  |     defaultTable = defaultTableName && getTable(defaultTableName); | ||||||
|  |     keywords = getKeywords(editor); | ||||||
|  |     identifierQuote = getIdentifierQuote(editor); | ||||||
|  |  | ||||||
|  |     if (defaultTableName && !defaultTable) | ||||||
|  |       defaultTable = findTableByAlias(defaultTableName, editor); | ||||||
|  |  | ||||||
|  |     defaultTable = defaultTable || []; | ||||||
|  |  | ||||||
|  |     if (defaultTable.columns) | ||||||
|  |       defaultTable = defaultTable.columns; | ||||||
|  |  | ||||||
|  |     var cur = editor.getCursor(); | ||||||
|  |     var result = []; | ||||||
|  |     var token = editor.getTokenAt(cur), start, end, search; | ||||||
|  |     if (token.end > cur.ch) { | ||||||
|  |       token.end = cur.ch; | ||||||
|  |       token.string = token.string.slice(0, cur.ch - token.start); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (token.string.match(/^[.`"\w@]\w*$/)) { | ||||||
|  |       search = token.string; | ||||||
|  |       start = token.start; | ||||||
|  |       end = token.end; | ||||||
|  |     } else { | ||||||
|  |       start = end = cur.ch; | ||||||
|  |       search = ""; | ||||||
|  |     } | ||||||
|  |     if (search.charAt(0) == "." || search.charAt(0) == identifierQuote) { | ||||||
|  |       start = nameCompletion(cur, token, result, editor); | ||||||
|  |     } else { | ||||||
|  |       var objectOrClass = function(w, className) { | ||||||
|  |         if (typeof w === "object") { | ||||||
|  |           w.className = className; | ||||||
|  |         } else { | ||||||
|  |           w = { text: w, className: className }; | ||||||
|  |         } | ||||||
|  |         return w; | ||||||
|  |       }; | ||||||
|  |     addMatches(result, search, defaultTable, function(w) { | ||||||
|  |         return objectOrClass(w, "CodeMirror-hint-table CodeMirror-hint-default-table"); | ||||||
|  |     }); | ||||||
|  |     addMatches( | ||||||
|  |         result, | ||||||
|  |         search, | ||||||
|  |         tables, function(w) { | ||||||
|  |           return objectOrClass(w, "CodeMirror-hint-table"); | ||||||
|  |         } | ||||||
|  |     ); | ||||||
|  |     if (!disableKeywords) | ||||||
|  |       addMatches(result, search, keywords, function(w) { | ||||||
|  |           return objectOrClass(w.toUpperCase(), "CodeMirror-hint-keyword"); | ||||||
|  |       }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |     return {list: result, from: Pos(cur.line, start), to: Pos(cur.line, end)}; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										123
									
								
								passbook/admin/static/codemirror/addon/hint/xml-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								passbook/admin/static/codemirror/addon/hint/xml-hint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,123 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var Pos = CodeMirror.Pos; | ||||||
|  |  | ||||||
|  |   function matches(hint, typed, matchInMiddle) { | ||||||
|  |     if (matchInMiddle) return hint.indexOf(typed) >= 0; | ||||||
|  |     else return hint.lastIndexOf(typed, 0) == 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function getHints(cm, options) { | ||||||
|  |     var tags = options && options.schemaInfo; | ||||||
|  |     var quote = (options && options.quoteChar) || '"'; | ||||||
|  |     var matchInMiddle = options && options.matchInMiddle; | ||||||
|  |     if (!tags) return; | ||||||
|  |     var cur = cm.getCursor(), token = cm.getTokenAt(cur); | ||||||
|  |     if (token.end > cur.ch) { | ||||||
|  |       token.end = cur.ch; | ||||||
|  |       token.string = token.string.slice(0, cur.ch - token.start); | ||||||
|  |     } | ||||||
|  |     var inner = CodeMirror.innerMode(cm.getMode(), token.state); | ||||||
|  |     if (!inner.mode.xmlCurrentTag) return | ||||||
|  |     var result = [], replaceToken = false, prefix; | ||||||
|  |     var tag = /\btag\b/.test(token.type) && !/>$/.test(token.string); | ||||||
|  |     var tagName = tag && /^\w/.test(token.string), tagStart; | ||||||
|  |  | ||||||
|  |     if (tagName) { | ||||||
|  |       var before = cm.getLine(cur.line).slice(Math.max(0, token.start - 2), token.start); | ||||||
|  |       var tagType = /<\/$/.test(before) ? "close" : /<$/.test(before) ? "open" : null; | ||||||
|  |       if (tagType) tagStart = token.start - (tagType == "close" ? 2 : 1); | ||||||
|  |     } else if (tag && token.string == "<") { | ||||||
|  |       tagType = "open"; | ||||||
|  |     } else if (tag && token.string == "</") { | ||||||
|  |       tagType = "close"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     var tagInfo = inner.mode.xmlCurrentTag(inner.state) | ||||||
|  |     if (!tag && !tagInfo || tagType) { | ||||||
|  |       if (tagName) | ||||||
|  |         prefix = token.string; | ||||||
|  |       replaceToken = tagType; | ||||||
|  |       var context = inner.mode.xmlCurrentContext ? inner.mode.xmlCurrentContext(inner.state) : [] | ||||||
|  |       var inner = context.length && context[context.length - 1] | ||||||
|  |       var curTag = inner && tags[inner] | ||||||
|  |       var childList = inner ? curTag && curTag.children : tags["!top"]; | ||||||
|  |       if (childList && tagType != "close") { | ||||||
|  |         for (var i = 0; i < childList.length; ++i) if (!prefix || matches(childList[i], prefix, matchInMiddle)) | ||||||
|  |           result.push("<" + childList[i]); | ||||||
|  |       } else if (tagType != "close") { | ||||||
|  |         for (var name in tags) | ||||||
|  |           if (tags.hasOwnProperty(name) && name != "!top" && name != "!attrs" && (!prefix || matches(name, prefix, matchInMiddle))) | ||||||
|  |             result.push("<" + name); | ||||||
|  |       } | ||||||
|  |       if (inner && (!prefix || tagType == "close" && matches(inner, prefix, matchInMiddle))) | ||||||
|  |         result.push("</" + inner + ">"); | ||||||
|  |     } else { | ||||||
|  |       // Attribute completion | ||||||
|  |       var curTag = tagInfo && tags[tagInfo.name], attrs = curTag && curTag.attrs; | ||||||
|  |       var globalAttrs = tags["!attrs"]; | ||||||
|  |       if (!attrs && !globalAttrs) return; | ||||||
|  |       if (!attrs) { | ||||||
|  |         attrs = globalAttrs; | ||||||
|  |       } else if (globalAttrs) { // Combine tag-local and global attributes | ||||||
|  |         var set = {}; | ||||||
|  |         for (var nm in globalAttrs) if (globalAttrs.hasOwnProperty(nm)) set[nm] = globalAttrs[nm]; | ||||||
|  |         for (var nm in attrs) if (attrs.hasOwnProperty(nm)) set[nm] = attrs[nm]; | ||||||
|  |         attrs = set; | ||||||
|  |       } | ||||||
|  |       if (token.type == "string" || token.string == "=") { // A value | ||||||
|  |         var before = cm.getRange(Pos(cur.line, Math.max(0, cur.ch - 60)), | ||||||
|  |                                  Pos(cur.line, token.type == "string" ? token.start : token.end)); | ||||||
|  |         var atName = before.match(/([^\s\u00a0=<>\"\']+)=$/), atValues; | ||||||
|  |         if (!atName || !attrs.hasOwnProperty(atName[1]) || !(atValues = attrs[atName[1]])) return; | ||||||
|  |         if (typeof atValues == 'function') atValues = atValues.call(this, cm); // Functions can be used to supply values for autocomplete widget | ||||||
|  |         if (token.type == "string") { | ||||||
|  |           prefix = token.string; | ||||||
|  |           var n = 0; | ||||||
|  |           if (/['"]/.test(token.string.charAt(0))) { | ||||||
|  |             quote = token.string.charAt(0); | ||||||
|  |             prefix = token.string.slice(1); | ||||||
|  |             n++; | ||||||
|  |           } | ||||||
|  |           var len = token.string.length; | ||||||
|  |           if (/['"]/.test(token.string.charAt(len - 1))) { | ||||||
|  |             quote = token.string.charAt(len - 1); | ||||||
|  |             prefix = token.string.substr(n, len - 2); | ||||||
|  |           } | ||||||
|  |           if (n) { // an opening quote | ||||||
|  |             var line = cm.getLine(cur.line); | ||||||
|  |             if (line.length > token.end && line.charAt(token.end) == quote) token.end++; // include a closing quote | ||||||
|  |           } | ||||||
|  |           replaceToken = true; | ||||||
|  |         } | ||||||
|  |         for (var i = 0; i < atValues.length; ++i) if (!prefix || matches(atValues[i], prefix, matchInMiddle)) | ||||||
|  |           result.push(quote + atValues[i] + quote); | ||||||
|  |       } else { // An attribute name | ||||||
|  |         if (token.type == "attribute") { | ||||||
|  |           prefix = token.string; | ||||||
|  |           replaceToken = true; | ||||||
|  |         } | ||||||
|  |         for (var attr in attrs) if (attrs.hasOwnProperty(attr) && (!prefix || matches(attr, prefix, matchInMiddle))) | ||||||
|  |           result.push(attr); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return { | ||||||
|  |       list: result, | ||||||
|  |       from: replaceToken ? Pos(cur.line, tagStart == null ? token.start : tagStart) : cur, | ||||||
|  |       to: replaceToken ? Pos(cur.line, token.end) : cur | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("hint", "xml", getHints); | ||||||
|  | }); | ||||||
							
								
								
									
										47
									
								
								passbook/admin/static/codemirror/addon/lint/coffeescript-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								passbook/admin/static/codemirror/addon/lint/coffeescript-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,47 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | // Depends on coffeelint.js from http://www.coffeelint.org/js/coffeelint.js | ||||||
|  |  | ||||||
|  | // declare global: coffeelint | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("lint", "coffeescript", function(text) { | ||||||
|  |   var found = []; | ||||||
|  |   if (!window.coffeelint) { | ||||||
|  |     if (window.console) { | ||||||
|  |       window.console.error("Error: window.coffeelint not defined, CodeMirror CoffeeScript linting cannot run."); | ||||||
|  |     } | ||||||
|  |     return found; | ||||||
|  |   } | ||||||
|  |   var parseError = function(err) { | ||||||
|  |     var loc = err.lineNumber; | ||||||
|  |     found.push({from: CodeMirror.Pos(loc-1, 0), | ||||||
|  |                 to: CodeMirror.Pos(loc, 0), | ||||||
|  |                 severity: err.level, | ||||||
|  |                 message: err.message}); | ||||||
|  |   }; | ||||||
|  |   try { | ||||||
|  |     var res = coffeelint.lint(text); | ||||||
|  |     for(var i = 0; i < res.length; i++) { | ||||||
|  |       parseError(res[i]); | ||||||
|  |     } | ||||||
|  |   } catch(e) { | ||||||
|  |     found.push({from: CodeMirror.Pos(e.location.first_line, 0), | ||||||
|  |                 to: CodeMirror.Pos(e.location.last_line, e.location.last_column), | ||||||
|  |                 severity: 'error', | ||||||
|  |                 message: e.message}); | ||||||
|  |   } | ||||||
|  |   return found; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										40
									
								
								passbook/admin/static/codemirror/addon/lint/css-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								passbook/admin/static/codemirror/addon/lint/css-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | // Depends on csslint.js from https://github.com/stubbornella/csslint | ||||||
|  |  | ||||||
|  | // declare global: CSSLint | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  | "use strict"; | ||||||
|  |  | ||||||
|  | CodeMirror.registerHelper("lint", "css", function(text, options) { | ||||||
|  |   var found = []; | ||||||
|  |   if (!window.CSSLint) { | ||||||
|  |     if (window.console) { | ||||||
|  |         window.console.error("Error: window.CSSLint not defined, CodeMirror CSS linting cannot run."); | ||||||
|  |     } | ||||||
|  |     return found; | ||||||
|  |   } | ||||||
|  |   var results = CSSLint.verify(text, options), messages = results.messages, message = null; | ||||||
|  |   for ( var i = 0; i < messages.length; i++) { | ||||||
|  |     message = messages[i]; | ||||||
|  |     var startLine = message.line -1, endLine = message.line -1, startCol = message.col -1, endCol = message.col; | ||||||
|  |     found.push({ | ||||||
|  |       from: CodeMirror.Pos(startLine, startCol), | ||||||
|  |       to: CodeMirror.Pos(endLine, endCol), | ||||||
|  |       message: message.message, | ||||||
|  |       severity : message.type | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |   return found; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | }); | ||||||
							
								
								
									
										59
									
								
								passbook/admin/static/codemirror/addon/lint/html-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								passbook/admin/static/codemirror/addon/lint/html-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | // Depends on htmlhint.js from http://htmlhint.com/js/htmlhint.js | ||||||
|  |  | ||||||
|  | // declare global: HTMLHint | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror"), require("htmlhint")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror", "htmlhint"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror, window.HTMLHint); | ||||||
|  | })(function(CodeMirror, HTMLHint) { | ||||||
|  |   "use strict"; | ||||||
|  |  | ||||||
|  |   var defaultRules = { | ||||||
|  |     "tagname-lowercase": true, | ||||||
|  |     "attr-lowercase": true, | ||||||
|  |     "attr-value-double-quotes": true, | ||||||
|  |     "doctype-first": false, | ||||||
|  |     "tag-pair": true, | ||||||
|  |     "spec-char-escape": true, | ||||||
|  |     "id-unique": true, | ||||||
|  |     "src-not-empty": true, | ||||||
|  |     "attr-no-duplication": true | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("lint", "html", function(text, options) { | ||||||
|  |     var found = []; | ||||||
|  |     if (HTMLHint && !HTMLHint.verify) { | ||||||
|  |       if(typeof HTMLHint.default !== 'undefined') { | ||||||
|  |         HTMLHint = HTMLHint.default; | ||||||
|  |       } else { | ||||||
|  |         HTMLHint = HTMLHint.HTMLHint; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (!HTMLHint) HTMLHint = window.HTMLHint; | ||||||
|  |     if (!HTMLHint) { | ||||||
|  |       if (window.console) { | ||||||
|  |           window.console.error("Error: HTMLHint not found, not defined on window, or not available through define/require, CodeMirror HTML linting cannot run."); | ||||||
|  |       } | ||||||
|  |       return found; | ||||||
|  |     } | ||||||
|  |     var messages = HTMLHint.verify(text, options && options.rules || defaultRules); | ||||||
|  |     for (var i = 0; i < messages.length; i++) { | ||||||
|  |       var message = messages[i]; | ||||||
|  |       var startLine = message.line - 1, endLine = message.line - 1, startCol = message.col - 1, endCol = message.col; | ||||||
|  |       found.push({ | ||||||
|  |         from: CodeMirror.Pos(startLine, startCol), | ||||||
|  |         to: CodeMirror.Pos(endLine, endCol), | ||||||
|  |         message: message.message, | ||||||
|  |         severity : message.type | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     return found; | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										63
									
								
								passbook/admin/static/codemirror/addon/lint/javascript-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								passbook/admin/static/codemirror/addon/lint/javascript-lint.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | // CodeMirror, copyright (c) by Marijn Haverbeke and others | ||||||
|  | // Distributed under an MIT license: https://codemirror.net/LICENSE | ||||||
|  |  | ||||||
|  | (function(mod) { | ||||||
|  |   if (typeof exports == "object" && typeof module == "object") // CommonJS | ||||||
|  |     mod(require("../../lib/codemirror")); | ||||||
|  |   else if (typeof define == "function" && define.amd) // AMD | ||||||
|  |     define(["../../lib/codemirror"], mod); | ||||||
|  |   else // Plain browser env | ||||||
|  |     mod(CodeMirror); | ||||||
|  | })(function(CodeMirror) { | ||||||
|  |   "use strict"; | ||||||
|  |   // declare global: JSHINT | ||||||
|  |  | ||||||
|  |   function validator(text, options) { | ||||||
|  |     if (!window.JSHINT) { | ||||||
|  |       if (window.console) { | ||||||
|  |         window.console.error("Error: window.JSHINT not defined, CodeMirror JavaScript linting cannot run."); | ||||||
|  |       } | ||||||
|  |       return []; | ||||||
|  |     } | ||||||
|  |     if (!options.indent) // JSHint error.character actually is a column index, this fixes underlining on lines using tabs for indentation | ||||||
|  |       options.indent = 1; // JSHint default value is 4 | ||||||
|  |     JSHINT(text, options, options.globals); | ||||||
|  |     var errors = JSHINT.data().errors, result = []; | ||||||
|  |     if (errors) parseErrors(errors, result); | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   CodeMirror.registerHelper("lint", "javascript", validator); | ||||||
|  |  | ||||||
|  |   function parseErrors(errors, output) { | ||||||
|  |     for ( var i = 0; i < errors.length; i++) { | ||||||
|  |       var error = errors[i]; | ||||||
|  |       if (error) { | ||||||
|  |         if (error.line <= 0) { | ||||||
|  |           if (window.console) { | ||||||
|  |             window.console.warn("Cannot display JSHint error (invalid line " + error.line + ")", error); | ||||||
|  |           } | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         var start = error.character - 1, end = start + 1; | ||||||
|  |         if (error.evidence) { | ||||||
|  |           var index = error.evidence.substring(start).search(/.\b/); | ||||||
|  |           if (index > -1) { | ||||||
|  |             end += index; | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Convert to format expected by validation service | ||||||
|  |         var hint = { | ||||||
|  |           message: error.reason, | ||||||
|  |           severity: error.code ? (error.code.startsWith('W') ? "warning" : "error") : "error", | ||||||
|  |           from: CodeMirror.Pos(error.line - 1, start), | ||||||
|  |           to: CodeMirror.Pos(error.line - 1, end) | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         output.push(hint); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }); | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	