Compare commits
	
		
			376 Commits
		
	
	
		
			version/0.
			...
			version/0.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 23146de2bf | |||
| e24f4fe3a8 | |||
| 8e6b69f96f | |||
| 979bea17ed | |||
| 30dba285d9 | |||
| 99fadf2e55 | |||
| b606e3d0cb | |||
| be642bc874 | |||
| 49a347b32f | |||
| 089b48aad1 | |||
| 2997cb83b1 | |||
| 08f0aca894 | |||
| 80ea7c40b7 | |||
| 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 | 
| @ -1,10 +1,10 @@ | ||||
| [bumpversion] | ||||
| current_version = 0.1.13-beta | ||||
| current_version = 0.7.4-beta | ||||
| tag = True | ||||
| commit = True | ||||
| parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)\-(?P<release>.*) | ||||
| serialize = {major}.{minor}.{patch}-{release} | ||||
| message = bump version: {current_version} -> {new_version} | ||||
| message = new release: {new_version} | ||||
| tag_name = version/{new_version} | ||||
|  | ||||
| [bumpversion:part:release] | ||||
| @ -23,29 +23,3 @@ values = | ||||
|  | ||||
| [bumpversion:file:passbook/__init__.py] | ||||
|  | ||||
| [bumpversion:file:passbook/api/__init__.py] | ||||
|  | ||||
| [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/hibp_policy/__init__.py] | ||||
|  | ||||
| [bumpversion:file:passbook/password_expiry_policy/__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] | ||||
| source = passbook | ||||
| omit = | ||||
|     env/ | ||||
|     */wsgi.py | ||||
|     manage.py | ||||
|     */migrations/* | ||||
|     */apps.py | ||||
|     passbook/management/commands/nexus_upload.py | ||||
|     passbook/management/commands/web.py | ||||
|     passbook/management/commands/worker.py | ||||
|     docs/ | ||||
| @ -23,6 +21,7 @@ exclude_lines = | ||||
|     def __str__ | ||||
|     def __repr__ | ||||
|     if self\.debug | ||||
|     if TYPE_CHECKING | ||||
|  | ||||
|     # Don't complain if tests don't hit defensive assertion code: | ||||
|     raise AssertionError | ||||
|  | ||||
| @ -2,3 +2,4 @@ env | ||||
| helm | ||||
| passbook-ui | ||||
| static | ||||
| *.env.yml | ||||
|  | ||||
| @ -9,3 +9,6 @@ insert_final_newline = true | ||||
|  | ||||
| [html] | ||||
| 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 | ||||
| /static/ | ||||
| local.env.yml | ||||
| .vscode/ | ||||
|  | ||||
| ### Helm ### | ||||
| # Chart dependencies | ||||
| **/charts/*.tgz | ||||
|  | ||||
							
								
								
									
										209
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							
							
						
						
									
										209
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							| @ -1,129 +1,164 @@ | ||||
| # Global Variables | ||||
| before_script: | ||||
|   - "python3 -m pip install -U virtualenv" | ||||
|   - "virtualenv env" | ||||
|   - "source env/bin/activate" | ||||
|   - "pip3 install -U -r requirements-dev.txt" | ||||
| stages: | ||||
|   - build-base-image | ||||
|   - build-dev-image | ||||
|   - test | ||||
|   - build | ||||
|   - docs | ||||
|   - deploy | ||||
| image: python:3.6 | ||||
| services: | ||||
|   - postgres:latest | ||||
|   - package | ||||
|   - post-release | ||||
| image: docker.beryju.org/passbook/dev:latest | ||||
|  | ||||
| variables: | ||||
|   POSTGRES_DB: passbook | ||||
|   POSTGRES_USER: passbook | ||||
|   POSTGRES_PASSWORD: 'EK-5jnKfjrGRm<77' | ||||
|   POSTGRES_PASSWORD: "EK-5jnKfjrGRm<77" | ||||
|  | ||||
| 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: | ||||
|   script: | ||||
|     - isort -c -sg env | ||||
|   stage: test | ||||
|   services: | ||||
|   - postgres:latest | ||||
|   - redis:latest | ||||
| migrations: | ||||
|   script: | ||||
|     - python manage.py migrate | ||||
|   stage: test | ||||
| prospector: | ||||
|   script: | ||||
|     - prospector | ||||
|   stage: test | ||||
|   services: | ||||
|   - postgres:latest | ||||
|   - redis:latest | ||||
| # prospector: | ||||
| #   script: | ||||
| #     - prospector | ||||
| #   stage: test | ||||
| #   services: | ||||
| #   - postgres:latest | ||||
| #   - redis:latest | ||||
| pylint: | ||||
|   script: | ||||
|     - pylint passbook | ||||
|   stage: test | ||||
|   services: | ||||
|   - postgres:latest | ||||
|   - redis:latest | ||||
| coverage: | ||||
|   script: | ||||
|     - coverage run manage.py test | ||||
|     - coverage run --concurrency=multiprocessing manage.py test | ||||
|     - coverage combine | ||||
|     - coverage report | ||||
|   stage: test | ||||
| bandit: | ||||
|   script: | ||||
|     - bandit -r passbook | ||||
|   stage: test | ||||
|   services: | ||||
|   - postgres:latest | ||||
|   - redis:latest | ||||
|  | ||||
| package-docker: | ||||
| build-passbook-server: | ||||
|   stage: build | ||||
|   image: | ||||
|     name: gcr.io/kaniko-project/executor:debug | ||||
|     entrypoint: [""] | ||||
|   before_script: | ||||
|     - echo "{\"auths\":{\"docker.$NEXUS_URL\":{\"auth\":\"$NEXUS_AUTH\"}}}" > /kaniko/.docker/config.json | ||||
|     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||
|   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.1.13-beta | ||||
|   stage: build | ||||
|     - /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.4-beta | ||||
|   only: | ||||
|     - tags | ||||
|     - /^version/.*$/ | ||||
| package-helm: | ||||
| build-passbook-static: | ||||
|   stage: build | ||||
|   script: | ||||
|     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||
|     - helm init --client-only | ||||
|     - helm package helm/passbook | ||||
|     - ./manage.py nexus_upload --method put --url $NEXUS_URL --auth $NEXUS_AUTH --repo helm *.tgz | ||||
|   only: | ||||
|     - tags | ||||
|     - /^version/.*$/ | ||||
| package-debian: | ||||
|   image: | ||||
|     name: gcr.io/kaniko-project/executor:debug | ||||
|     entrypoint: [""] | ||||
|   before_script: | ||||
|     - apt update | ||||
|     - apt install -y --no-install-recommends build-essential debhelper devscripts equivs python3 python3-dev python3-pip libsasl2-dev libldap2-dev | ||||
|     - mk-build-deps debian/control | ||||
|     - apt install ./*build-deps*deb -f -y | ||||
|     - python3 -m pip install -U virtualenv pip | ||||
|     - virtualenv env | ||||
|     - source env/bin/activate | ||||
|     - pip3 install -U -r requirements.txt -r requirements-dev.txt | ||||
|     - ./manage.py collectstatic --no-input | ||||
|   image: ubuntu:18.04 | ||||
|     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||
|   script: | ||||
|     - debuild -us -uc | ||||
|     - cp ../passbook*.deb . | ||||
|     - ./manage.py nexus_upload --method post --url $NEXUS_URL --auth $NEXUS_AUTH --repo apt passbook*deb | ||||
|     - /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.4-beta | ||||
|   only: | ||||
|     - tags | ||||
|     - /^version/.*$/ | ||||
|   # running collectstatic fully initialises django, hence we need that databases | ||||
|   services: | ||||
|     - postgres:latest | ||||
|     - redis:latest | ||||
| # build-passbook-gatekeeper: | ||||
| #   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: | ||||
| #     - /kaniko/executor --context $CI_PROJECT_DIR/gatekeeper --dockerfile $CI_PROJECT_DIR/gatekeeper/Dockerfile --destination docker.beryju.org/passbook/gatekeeper:latest --destination docker.beryju.org/passbook/gatekeeper:0.7.4-beta | ||||
| #   only: | ||||
| #     - tags | ||||
| #     - /^version/.*$/ | ||||
|  | ||||
| package-helm: | ||||
|   image: debian:stretch-slim | ||||
|   stage: package | ||||
|   before_script: | ||||
|     - apt update && apt install -y curl | ||||
|     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||
|   script: | ||||
|     - helm init --client-only | ||||
|     - helm dependency update helm/passbook | ||||
|     - helm package helm/passbook | ||||
|   artifacts: | ||||
|     paths: | ||||
|     - passbook*deb | ||||
|     expire_in: 2 days | ||||
|   stage: build | ||||
|       - passbook-*.tgz | ||||
|     expire_in: 1 week | ||||
|   only: | ||||
|   - tags | ||||
|   - /^version/.*$/ | ||||
|     - tags | ||||
|     - /^version/.*$/ | ||||
|  | ||||
| # docs: | ||||
| #   stage: docs | ||||
| #   only: | ||||
| #   - master | ||||
| #   - tags | ||||
| #   - /^debian/.*$/ | ||||
| #   environment: | ||||
| #     name: docs | ||||
| #     url: "https://passbook.beryju.org/docs/" | ||||
| #   script: | ||||
| #     - apt update | ||||
| #     - apt install -y rsync | ||||
| #     - "mkdir ~/.ssh" | ||||
| #     - "cp .gitlab/known_hosts ~/.ssh/" | ||||
| #     - "pip3 install -U -r requirements-docs.txt" | ||||
| #     - "eval $(ssh-agent -s)" | ||||
| #     - "echo \"${CI_SSH_PRIVATE}\" | ssh-add -" | ||||
| #     - mkdocs build | ||||
| #     - 'rsync -avh --delete web/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/"' | ||||
| #     - 'rsync -avh --delete site/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/docs/"' | ||||
|  | ||||
| # deploy: | ||||
| #   environment: | ||||
| #     name: production | ||||
| #     url: https://passbook-prod.default.k8s.beryju.org/ | ||||
| #   stage: deploy | ||||
| #   only: | ||||
| #   - tags | ||||
| #   - /^version/.*$/ | ||||
| #   script: | ||||
| #     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||
| #     - helm init | ||||
| #     - helm upgrade passbook-prod helm/passbook --devel | ||||
| notify-sentry: | ||||
|   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.4-beta | ||||
|     - sentry-cli releases set-commits --auto passbook@0.7.4-beta | ||||
|   only: | ||||
|     - tags | ||||
|     - /^version/.*$/ | ||||
|  | ||||
| @ -3,7 +3,6 @@ test-warnings: true | ||||
| doc-warnings: false | ||||
|  | ||||
| ignore-paths: | ||||
|   - env | ||||
|   - migrations | ||||
|   - docs | ||||
|   - 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 | ||||
| load-plugins=pylint_django,pylint.extensions.bad_builtin | ||||
| #,pylint.extensions.docparams | ||||
| extension-pkg-whitelist=lxml | ||||
| const-rgx=[a-zA-Z0-9_]{1,40}$ | ||||
| ignored-modules=django-otp | ||||
| jobs=4 | ||||
|  | ||||
| [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" | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										29
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -1,34 +1,9 @@ | ||||
| FROM python:3.6-slim-stretch as build | ||||
| FROM docker.beryju.org/passbook/base:latest | ||||
|  | ||||
| COPY ./passbook/ /app/passbook | ||||
| COPY ./manage.py /app/ | ||||
| COPY ./requirements.txt /app/ | ||||
| COPY ./docker/uwsgi.ini /app/ | ||||
|  | ||||
| WORKDIR /app/ | ||||
|  | ||||
| RUN apt-get update && apt-get install build-essential libssl-dev libffi-dev -y && \ | ||||
|     mkdir /app/static/ && \ | ||||
|     pip install -r requirements.txt && \ | ||||
|     pip install psycopg2 && \ | ||||
|     ./manage.py collectstatic --no-input && \ | ||||
|     apt-get remove --purge -y build-essential && \ | ||||
|     apt-get autoremove --purge -y | ||||
|  | ||||
| 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 apt-get update && apt-get install build-essential libssl-dev libffi-dev -y && \ | ||||
|     pip install -r requirements.txt && \ | ||||
|     pip install psycopg2 && \ | ||||
|     adduser --system --home /app/ passbook && \ | ||||
|     chown -R passbook /app/ && \ | ||||
|     apt-get remove --purge -y build-essential && \ | ||||
|     apt-get autoremove --purge -y | ||||
|  | ||||
| USER passbook | ||||
|  | ||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @ -1,6 +1,6 @@ | ||||
| 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 | ||||
| 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
											
										
									
								
							
							
								
								
									
										13
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | ||||
| # passbook | ||||
|  | ||||
| ## Quick instance | ||||
|  | ||||
| ``` | ||||
| export PASSBOOK_DOMAIN=domain.tld | ||||
| # Optionally enable Error-reporting | ||||
| # export PASSBOOK_ERROR_REPORTING=true | ||||
| 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 | ||||
							
								
								
									
										71
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										71
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							| @ -1,71 +0,0 @@ | ||||
| passbook (0.1.13) stable; urgency=medium | ||||
|  | ||||
|   * bump version: 0.1.11-beta -> 0.1.12-beta | ||||
|   * Fix DoesNotExist error when running PolicyEngine against None user | ||||
|   * allow custom email server for helm installs | ||||
|   * fix UserChangePasswordView not requiring Login | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Mon, 11 Mar 2019 10:28:36 +0000 | ||||
|  | ||||
| passbook (0.1.12) stable; urgency=medium | ||||
|  | ||||
|   * bump version: 0.1.10-beta -> 0.1.11-beta | ||||
|   * rewrite PasswordFactor to use backends setting instead of trying all backends | ||||
|   * install updated helm release from local folder | ||||
|   * disable automatic k8s deployment for now | ||||
|   * fix OAuth Authorization View not requiring authentication | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Mon, 11 Mar 2019 08:50:29 +0000 | ||||
|  | ||||
| passbook (0.1.11) stable; urgency=medium | ||||
|  | ||||
|   * add group administration | ||||
|   * bump version: 0.1.9-beta -> 0.1.10-beta | ||||
|   * fix helm labels being on deployments and not pods | ||||
|   * automatically deploy after release | ||||
|   * use Django's Admin FilteredSelectMultiple for Group Membership | ||||
|   * always use FilteredSelectMultiple for many-to-many fields | ||||
|   * Add Group Member policy | ||||
|   * add LDAP Group Membership Policy | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Sun, 10 Mar 2019 18:55:31 +0000 | ||||
|  | ||||
| passbook (0.1.10) stable; urgency=high | ||||
|  | ||||
|   * bump version: 0.1.7-beta -> 0.1.8-beta | ||||
|   * consistently using PolicyEngine | ||||
|   * add more Verbosity to PolicyEngine, rewrite SAML Authorisation check | ||||
|   * slightly refactor Factor View, add more unittests | ||||
|   * add impersonation middleware, add to templates | ||||
|   * bump version: 0.1.8-beta -> 0.1.9-beta | ||||
|   * fix k8s service routing http traffic to workers | ||||
|   * Fix button on policy test page | ||||
|   * better show loading state when testing a policy | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Sun, 10 Mar 2019 14:52:40 +0000 | ||||
|  | ||||
| passbook (0.1.7) stable; urgency=medium | ||||
|  | ||||
|   * bump version: 0.1.3-beta -> 0.1.4-beta | ||||
|   * implicitly add kubernetes-healthcheck-host in helm configmap | ||||
|   * fix debian build (again) | ||||
|   * add PropertyMapping Model, add Subclass for SAML, test with AWS | ||||
|   * add custom DynamicArrayField to better handle arrays | ||||
|   * format data before inserting it | ||||
|   * bump version: 0.1.4-beta -> 0.1.5-beta | ||||
|   * fix static files missing for debian package | ||||
|   * fix password not getting set on user import | ||||
|   * remove audit's login attempt | ||||
|   * add passing property to PolicyEngine | ||||
|   * fix captcha factor not loading keys from Factor class | ||||
|   * bump version: 0.1.5-beta -> 0.1.6-beta | ||||
|   * fix MATCH_EXACT not working as intended | ||||
|   * Improve access control for saml | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Fri, 08 Mar 2019 20:37:05 +0000 | ||||
|  | ||||
| passbook (0.1.4) stable; urgency=medium | ||||
|  | ||||
|   * initial debian package release | ||||
|  | ||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Wed, 06 Mar 2019 18:22:41 +0000 | ||||
							
								
								
									
										1
									
								
								debian/compat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/compat
									
									
									
									
										vendored
									
									
								
							| @ -1 +0,0 @@ | ||||
| 10 | ||||
							
								
								
									
										20
									
								
								debian/config
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								debian/config
									
									
									
									
										vendored
									
									
								
							| @ -1,20 +0,0 @@ | ||||
| #!/bin/sh | ||||
| # config maintainer script for passbook | ||||
| set -e | ||||
|  | ||||
| # source debconf stuff | ||||
| . /usr/share/debconf/confmodule | ||||
|  | ||||
| dbc_first_version=1.0.0 | ||||
| dbc_dbuser=passbook | ||||
| dbc_dbname=passbook | ||||
|  | ||||
| # source dbconfig-common shell library, and call the hook function | ||||
| if [ -f /usr/share/dbconfig-common/dpkg/config.pgsql ]; then | ||||
|     . /usr/share/dbconfig-common/dpkg/config.pgsql | ||||
|     dbc_go passbook "$@" | ||||
| fi | ||||
|  | ||||
| #DEBHELPER# | ||||
|  | ||||
| exit 0 | ||||
							
								
								
									
										14
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | ||||
| Source: passbook | ||||
| Section: admin | ||||
| Priority: optional | ||||
| Maintainer: BeryJu.org <support@beryju.org> | ||||
| Uploaders: Jens Langhammer <jens@beryju.org>, BeryJu.org <support@beryju.org> | ||||
| Build-Depends: debhelper (>= 10), dh-systemd (>= 1.5), dh-exec, wget, dh-exec, python3 (>= 3.5) | python3.6 | python3.7 | ||||
| Standards-Version: 3.9.6 | ||||
|  | ||||
| Package: passbook | ||||
| Architecture: all | ||||
| Recommends: mysql-server, redis-server | ||||
| Pre-Depends: adduser, libldap2-dev, libsasl2-dev | ||||
| Depends: python3 (>= 3.5) | python3.6 | python3.7, python3-pip, dbconfig-pgsql | dbconfig-no-thanks, ${misc:Depends} | ||||
| Description: Authentication Provider/Proxy supporting protocols like SAML, OAuth, LDAP and more. | ||||
							
								
								
									
										22
									
								
								debian/copyright
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								debian/copyright
									
									
									
									
										vendored
									
									
								
							| @ -1,22 +0,0 @@ | ||||
| MIT License | ||||
|  | ||||
| Copyright (c) 2019 BeryJu.org | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in all | ||||
| copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
| SOFTWARE. | ||||
|  | ||||
							
								
								
									
										4
									
								
								debian/dirs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								debian/dirs
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +0,0 @@ | ||||
| etc/passbook/ | ||||
| etc/passbook/config.d/ | ||||
| var/log/passbook/ | ||||
| usr/share/passbook/ | ||||
							
								
								
									
										44
									
								
								debian/etc/passbook/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										44
									
								
								debian/etc/passbook/config.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,44 +0,0 @@ | ||||
| debug: false | ||||
| http: | ||||
|     host: 0.0.0.0 | ||||
|     port: 8000 | ||||
| secret_key_file: /etc/passbook/secret_key | ||||
| log: | ||||
|     level: | ||||
|         console: INFO | ||||
|         file: DEBUG | ||||
|     file: /var/log/passbook/passbook.log | ||||
| # Error reporting, disabled by default | ||||
| # error_report_enabled: true | ||||
|  | ||||
| # Set this to the server's external address. | ||||
| # This is used to generate external URLs | ||||
| external_url: http://image.example.com | ||||
|  | ||||
| # This dictates how the Path is generated | ||||
| # can be either of: | ||||
| # - view_sha512_short | ||||
| # - view_md5 | ||||
| # - view_sha256 | ||||
| # - view_sha512 | ||||
| default_return_view: view_sha256 | ||||
|  | ||||
| # Set this to true if you only want to use external authentication | ||||
| external_auth_only: false | ||||
|  | ||||
| # If this is true, images are automatically claimed if the windows user exists | ||||
| # in django | ||||
| auto_claim_enabled: true | ||||
|  | ||||
| # LDAP Authentication | ||||
| # ldap: | ||||
| #     enabled: false | ||||
| #     server: | ||||
| #         uri: 'ldap://dc1.example.com' | ||||
| #         tls: false | ||||
| #     bind: | ||||
| #         dn: '' | ||||
| #         password: '' | ||||
| #     search_base: '' | ||||
| #     filter: '(sAMAccountName=%(user)s)' | ||||
| #     require_group: '' | ||||
							
								
								
									
										2
									
								
								debian/gbp.conf
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								debian/gbp.conf
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +0,0 @@ | ||||
| [buildpackage] | ||||
| export-dir=../build-area | ||||
							
								
								
									
										8
									
								
								debian/install
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								debian/install
									
									
									
									
										vendored
									
									
								
							| @ -1,8 +0,0 @@ | ||||
| passbook		/usr/share/passbook/ | ||||
| static			/usr/share/passbook/ | ||||
| manage.py		/usr/share/passbook/ | ||||
| passbook.sh		/usr/share/passbook/ | ||||
| vendor			/usr/share/passbook/ | ||||
|  | ||||
| debian/etc/passbook								/etc/ | ||||
| debian/templates/database.yml					/usr/share/passbook/ | ||||
							
								
								
									
										14
									
								
								debian/passbook-worker.service
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/passbook-worker.service
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | ||||
| [Unit] | ||||
| Description=passbook - Authentication Provider/Proxy (Background worker) | ||||
| After=network.target | ||||
| Requires=network.target | ||||
|  | ||||
| [Service] | ||||
| User=passbook | ||||
| Group=passbook | ||||
| WorkingDirectory=/usr/share/passbook | ||||
| Type=simple | ||||
| ExecStart=/usr/share/passbook/passbook.sh worker | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
							
								
								
									
										14
									
								
								debian/passbook.service
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/passbook.service
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | ||||
| [Unit] | ||||
| Description=passbook - Authentication Provider/Proxy | ||||
| After=network.target | ||||
| Requires=network.target | ||||
|  | ||||
| [Service] | ||||
| User=passbook | ||||
| Group=passbook | ||||
| WorkingDirectory=/usr/share/passbook | ||||
| Type=simple | ||||
| ExecStart=/usr/share/passbook/passbook.sh web | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
							
								
								
									
										36
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							| @ -1,36 +0,0 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| set -e | ||||
|  | ||||
| . /usr/share/debconf/confmodule | ||||
| . /usr/share/dbconfig-common/dpkg/postinst.pgsql | ||||
|  | ||||
| # you can set the default database encoding to something else | ||||
| dbc_pgsql_createdb_encoding="UTF8" | ||||
| dbc_generate_include=template:/etc/passbook/config.d/database.yml | ||||
| dbc_generate_include_args="-o template_infile=/usr/share/passbook/database.yml" | ||||
| dbc_go passbook "$@" | ||||
|  | ||||
| if [ -z "`getent group passbook`" ]; then | ||||
| 	addgroup --quiet --system passbook | ||||
| fi | ||||
| if [ -z "`getent passwd passbook`" ]; then | ||||
| 	echo " * Creating user and group passbook..." | ||||
| 	adduser --quiet --system --home /usr/share/passbook --shell /bin/false --ingroup passbook --disabled-password --disabled-login --gecos "passbook User" passbook  >> /var/log/passbook/passbook.log 2>&1 | ||||
| fi | ||||
| echo " * Updating binary packages (psycopg2)" | ||||
| python3 -m pip install --target=/usr/share/passbook/vendor/ --no-cache-dir --upgrade --force-reinstall psycopg2 >> /var/log/passbook/passbook.log 2>&1 | ||||
| if [ ! -f '/etc/passbook/secret_key' ]; then | ||||
| 	echo " * Generating Secret Key" | ||||
| 	python3 -c 'import random; result = "".join([random.choice("abcdefghijklmnopqrstuvwxyz0123456789!@#$%^&*(-_=+)") for i in range(50)]); print(result)' > /etc/passbook/secret_key 2> /dev/null | ||||
| fi | ||||
| chown -R passbook: /usr/share/passbook/ | ||||
| chown -R passbook: /etc/passbook/ | ||||
| chown -R passbook: /var/log/passbook/ | ||||
| chmod 440 /etc/passbook/secret_key | ||||
| echo " * Running Database Migration" | ||||
| /usr/share/passbook/passbook.sh migrate | ||||
| echo " * A superuser can be created with this command '/usr/share/passbook/passbook.sh createsuperuser'" | ||||
| echo " * You should probably also adjust your settings in '/etc/passbook/config.yml'" | ||||
|  | ||||
| #DEBHELPER# | ||||
							
								
								
									
										24
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							| @ -1,24 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| set -e | ||||
|  | ||||
| if [ -f /usr/share/debconf/confmodule ]; then | ||||
|     . /usr/share/debconf/confmodule | ||||
| fi | ||||
| if [ -f /usr/share/dbconfig-common/dpkg/postrm.pgsql ]; then | ||||
|     . /usr/share/dbconfig-common/dpkg/postrm.pgsql | ||||
|     dbc_go passbook "$@" | ||||
| fi | ||||
|  | ||||
|  | ||||
| if [ "$1" = "purge" ]; then | ||||
|     if which ucf >/dev/null 2>&1; then | ||||
|         ucf --purge /etc/passbook/config.d/database.yml | ||||
|         ucfr --purge passbook /etc/passbook/config.d/database.yml | ||||
|     fi | ||||
|     rm -rf /etc/passbook/ | ||||
|     rm -rf /usr/share/passbook/ | ||||
| fi | ||||
|  | ||||
| #DEBHELPER# | ||||
|  | ||||
							
								
								
									
										10
									
								
								debian/prerm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								debian/prerm
									
									
									
									
										vendored
									
									
								
							| @ -1,10 +0,0 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| set -e | ||||
|  | ||||
| . /usr/share/debconf/confmodule | ||||
| . /usr/share/dbconfig-common/dpkg/prerm.pgsql | ||||
| dbc_go passbook "$@" | ||||
|  | ||||
| #DEBHELPER# | ||||
|  | ||||
							
								
								
									
										27
									
								
								debian/rules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								debian/rules
									
									
									
									
										vendored
									
									
								
							| @ -1,27 +0,0 @@ | ||||
| #!/usr/bin/make -f | ||||
|  | ||||
| # Uncomment this to turn on verbose mode. | ||||
| # export DH_VERBOSE=1 | ||||
|  | ||||
| %: | ||||
| 	dh $@ --with=systemd | ||||
|  | ||||
| build-arch: | ||||
| 	python3 -m pip install setuptools | ||||
| 	python3 -m pip install --target=vendor/ -r requirements.txt | ||||
|  | ||||
| override_dh_strip: | ||||
| 	dh_strip --exclude=psycopg2 | ||||
|  | ||||
| override_dh_shlibdeps: | ||||
| 	dh_shlibdeps --exclude=psycopg2 | ||||
|  | ||||
| override_dh_installinit: | ||||
| 	dh_installinit --name=passbook | ||||
| 	dh_installinit --name=passbook-worker | ||||
| 	dh_systemd_enable --name=passbook | ||||
| 	dh_systemd_enable --name=passbook-worker | ||||
| 	dh_systemd_start | ||||
|  | ||||
| # override_dh_usrlocal to do nothing | ||||
| override_dh_usrlocal: | ||||
							
								
								
									
										1
									
								
								debian/source/format
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/source/format
									
									
									
									
										vendored
									
									
								
							| @ -1 +0,0 @@ | ||||
| 3.0 (native) | ||||
							
								
								
									
										8
									
								
								debian/templates/database.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								debian/templates/database.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,8 +0,0 @@ | ||||
| databases: | ||||
|   default: | ||||
|     engine: django.db.backends.postgresql | ||||
|     name: _DBC_DBNAME_ | ||||
|     user: _DBC_DBUSER_ | ||||
|     password: _DBC_DBPASS_ | ||||
|     host: _DBC_DBSERVER_ | ||||
|     port: _DBC_DBPORT_ | ||||
							
								
								
									
										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 | ||||
							
								
								
									
										87
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,87 @@ | ||||
| --- | ||||
| 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: | ||||
|     image: docker.beryju.org/passbook/server:${SERVER_TAG:-latest} | ||||
|     command: | ||||
|       - uwsgi | ||||
|       - uwsgi.ini | ||||
|     environment: | ||||
|       - PASSBOOK_DOMAIN=${PASSBOOK_DOMAIN} | ||||
|       - PASSBOOK_REDIS__HOST=redis | ||||
|       - PASSBOOK_ERROR_REPORTING=${PASSBOOK_ERROR_REPORTING:-false} | ||||
|       - 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_ERROR_REPORTING=${PASSBOOK_ERROR_REPORTING:-false} | ||||
|       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||
|       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||
|   static: | ||||
|     image: docker.beryju.org/passbook/static:latest | ||||
|     networks: | ||||
|       - internal | ||||
|     labels: | ||||
|       - traefik.frontend.rule=PathPrefix:/static, /robots.txt | ||||
|       - traefik.port=8080 | ||||
|       - 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: {} | ||||
							
								
								
									
										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 | ||||
							
								
								
									
										8
									
								
								gatekeeper/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								gatekeeper/Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | ||||
| FROM quay.io/pusher/oauth2_proxy | ||||
|  | ||||
| COPY templates /templates | ||||
|  | ||||
| ENV OAUTH2_PROXY_EMAIL_DOMAINS=* | ||||
| ENV OAUTH2_PROXY_PROVIDER=oidc | ||||
| ENV OAUTH2_PROXY_CUSTOM_TEMPLATES_DIR=/templates | ||||
| ENV OAUTH2_PROXY_HTTP_ADDRESS=:4180 | ||||
							
								
								
									
										18
									
								
								gatekeeper/templates/error.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								gatekeeper/templates/error.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | ||||
| {{define "error.html"}} | ||||
| <!DOCTYPE html> | ||||
| <html lang="en" charset="utf-8"> | ||||
|  | ||||
| <head> | ||||
|     <title>{{.Title}}</title> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no"> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     <h2>{{.Title}}</h2> | ||||
|     <p>{{.Message}}</p> | ||||
|     <hr> | ||||
|     <p><a href="{{.ProxyPrefix}}/sign_in">Sign In</a></p> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
| {{end}} | ||||
							
								
								
									
										119
									
								
								gatekeeper/templates/sign_in.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										119
									
								
								gatekeeper/templates/sign_in.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,119 @@ | ||||
| {{define "sign_in.html"}} | ||||
| <!DOCTYPE html> | ||||
| <html lang="en" charset="utf-8"> | ||||
| <head> | ||||
|     <title>Sign In with passbook</title> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no"> | ||||
|     <style> | ||||
|         body { | ||||
|             font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; | ||||
|             font-size: 14px; | ||||
|             line-height: 1.42857143; | ||||
|             color: #333; | ||||
|             background: #f0f0f0; | ||||
|         } | ||||
|  | ||||
|         .signin { | ||||
|             display: block; | ||||
|             margin: 20px auto; | ||||
|             max-width: 400px; | ||||
|             background: #fff; | ||||
|             border: 1px solid #ccc; | ||||
|             border-radius: 10px; | ||||
|             padding: 20px; | ||||
|         } | ||||
|  | ||||
|         .center { | ||||
|             text-align: center; | ||||
|         } | ||||
|  | ||||
|         .btn { | ||||
|             color: #fff; | ||||
|             background-color: #428bca; | ||||
|             border: 1px solid #357ebd; | ||||
|             -webkit-border-radius: 4; | ||||
|             -moz-border-radius: 4; | ||||
|             border-radius: 4px; | ||||
|             font-size: 14px; | ||||
|             padding: 6px 12px; | ||||
|             text-decoration: none; | ||||
|             cursor: pointer; | ||||
|         } | ||||
|  | ||||
|         .btn:hover { | ||||
|             background-color: #3071a9; | ||||
|             border-color: #285e8e; | ||||
|             text-decoration: none; | ||||
|         } | ||||
|  | ||||
|         label { | ||||
|             display: inline-block; | ||||
|             max-width: 100%; | ||||
|             margin-bottom: 5px; | ||||
|             font-weight: 700; | ||||
|         } | ||||
|  | ||||
|         input { | ||||
|             display: block; | ||||
|             width: 100%; | ||||
|             height: 34px; | ||||
|             padding: 6px 12px; | ||||
|             font-size: 14px; | ||||
|             line-height: 1.42857143; | ||||
|             color: #555; | ||||
|             background-color: #fff; | ||||
|             background-image: none; | ||||
|             border: 1px solid #ccc; | ||||
|             border-radius: 4px; | ||||
|             -webkit-box-shadow: inset 0 1px 1px rgba(0, 0, 0, .075); | ||||
|             box-shadow: inset 0 1px 1px rgba(0, 0, 0, .075); | ||||
|             -webkit-transition: border-color ease-in-out .15s, -webkit-box-shadow ease-in-out .15s; | ||||
|             -o-transition: border-color ease-in-out .15s, box-shadow ease-in-out .15s; | ||||
|             transition: border-color ease-in-out .15s, box-shadow ease-in-out .15s; | ||||
|             margin: 0; | ||||
|             box-sizing: border-box; | ||||
|         } | ||||
|  | ||||
|         footer { | ||||
|             display: block; | ||||
|             font-size: 10px; | ||||
|             color: #aaa; | ||||
|             text-align: center; | ||||
|             margin-bottom: 10px; | ||||
|         } | ||||
|  | ||||
|         footer a { | ||||
|             display: inline-block; | ||||
|             height: 25px; | ||||
|             line-height: 25px; | ||||
|             color: #aaa; | ||||
|             text-decoration: underline; | ||||
|         } | ||||
|  | ||||
|         footer a:hover { | ||||
|             color: #aaa; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
|  | ||||
| <body> | ||||
|     <div class="signin center"> | ||||
|         <form method="GET" action="{{.ProxyPrefix}}/start"> | ||||
|             <input type="hidden" name="rd" value="{{.Redirect}}"> | ||||
|             <button type="submit" class="btn">Sign in with passbook</button><br /> | ||||
|         </form> | ||||
|     </div> | ||||
|     <script> | ||||
|         if (window.location.hash) { | ||||
|             (function () { | ||||
|                 var inputs = document.getElementsByName('rd'); | ||||
|                 for (var i = 0; i < inputs.length; i++) { | ||||
|                     inputs[i].value += window.location.hash; | ||||
|                 } | ||||
|             })(); | ||||
|         } | ||||
|     </script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
| {{end}} | ||||
							
								
								
									
										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 | ||||
| appVersion: "0.1.13-beta" | ||||
| appVersion: "0.7.4-beta" | ||||
| description: A Helm chart for passbook. | ||||
| name: passbook | ||||
| version: "0.1.13-beta" | ||||
| icon: https://passbook.beryju.org/images/logo.png | ||||
| version: "0.7.4-beta" | ||||
| 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: | ||||
| - name: redis | ||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||
|   version: 5.1.0 | ||||
| - name: postgresql | ||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||
|   version: 3.10.1 | ||||
| digest: sha256:04bd136761f070e94a2ff32ff48ff87f5e07fbd451e5fd7f65551e3bd4680e5e | ||||
| generated: 2019-02-08T12:08:49.090666+01:00 | ||||
|   version: 6.5.8 | ||||
| - name: redis | ||||
|   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: | ||||
| - name: redis | ||||
|   version: 5.1.0 | ||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||
| - 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/ | ||||
|  | ||||
							
								
								
									
										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_reporting: {{ .Values.config.error_reporting }} | ||||
|     domain: ".{{ index .Values.ingress.hosts 0 }}" | ||||
| @ -1,6 +1,5 @@ | ||||
| {{- if .Values.ingress.enabled -}} | ||||
| {{- $fullName := include "passbook.fullname" . -}} | ||||
| {{- $ingressPath := .Values.ingress.path -}} | ||||
| apiVersion: extensions/v1beta1 | ||||
| kind: Ingress | ||||
| metadata: | ||||
| @ -30,9 +29,17 @@ spec: | ||||
|     - host: {{ . | quote }} | ||||
|       http: | ||||
|         paths: | ||||
|           - path: {{ $ingressPath }} | ||||
|           - path: / | ||||
|             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 | ||||
|   {{- end }} | ||||
| {{- end }} | ||||
| @ -1,139 +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: {{ .Values.config.email.host }} | ||||
|       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: ":{{ .Values.redis.password }}@{{ .Release.Name }}-redis-master" | ||||
|     # 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 }} | ||||
|         - kubernetes-healthcheck-host | ||||
|  | ||||
|     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: | ||||
|           username: "%(sAMAccountName)s" | ||||
|           email: "%(mail)s" | ||||
|           name: "%(displayName)" | ||||
|       # # 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.aws | ||||
|         - 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,67 +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 }} | ||||
|         passbook.io/component: web | ||||
|     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,52 +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 }} | ||||
|         passbook.io/component: worker | ||||
|     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 }} | ||||
							
								
								
									
										121
									
								
								helm/passbook/templates/prom-rules.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								helm/passbook/templates/prom-rules.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,121 @@ | ||||
| {{- 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 | ||||
| {{- 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: bW9uaXRvcg== # monitor in base64 | ||||
|   {{- if .Values.config.secret_key }} | ||||
|   secret_key: {{ .Values.config.secret_key | b64enc | quote }} | ||||
|   {{- else }} | ||||
|   secret_key: {{ randAlphaNum 50 | b64enc | quote}} | ||||
|   {{- end }} | ||||
							
								
								
									
										48
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | ||||
| 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: 8080 | ||||
|               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 | ||||
							
								
								
									
										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: 8080 | ||||
|       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,16 +1,17 @@ | ||||
| apiVersion: v1 | ||||
| kind: Service | ||||
| metadata: | ||||
|   name: {{ include "passbook.fullname" . }} | ||||
|   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 }} | ||||
|     helm.sh/chart: {{ include "passbook.chart" . }} | ||||
|     k8s.passbook.io/component: web | ||||
| spec: | ||||
|   type: {{ .Values.service.type }} | ||||
|   type: ClusterIP | ||||
|   ports: | ||||
|     - port: {{ .Values.service.port }} | ||||
|     - port: 80 | ||||
|       targetPort: http | ||||
|       protocol: TCP | ||||
|       name: http | ||||
							
								
								
									
										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. | ||||
| # This is a YAML-formatted file. | ||||
| # Declare variables to be passed into your templates. | ||||
|  | ||||
| replicaCount: 1 | ||||
|  | ||||
| image: | ||||
|   tag: 0.1.13-beta | ||||
|   tag: 0.7.4-beta | ||||
|  | ||||
| nameOverride: "" | ||||
|  | ||||
| @ -13,17 +10,14 @@ config: | ||||
|   # Optionally specify fixed secret_key, otherwise generated automatically | ||||
|   # secret_key: _k*@6h2u2@q-dku57hhgzb7tnx*ba9wodcb^s9g0j59@=y(@_o | ||||
|   # Enable error reporting | ||||
|   error_reporting: true | ||||
|   error_reporting: false | ||||
|   email: | ||||
|     host: localhost | ||||
|  | ||||
| postgresql: | ||||
|     postgresqlDatabase: passbook | ||||
|     postgresqlPassword: foo | ||||
|  | ||||
| service: | ||||
|   type: ClusterIP | ||||
|   port: 80 | ||||
| # This Helm chart ships with built-in Prometheus ServiceMonitors and Rules. | ||||
| # This requires the CoreOS Prometheus Operator. | ||||
| monitoring: | ||||
|   enabled: false | ||||
|  | ||||
| ingress: | ||||
|   enabled: false | ||||
| @ -33,27 +27,18 @@ ingress: | ||||
|   path: / | ||||
|   hosts: | ||||
|     - passbook.k8s.local | ||||
|     - kubernetes-healthcheck-host | ||||
|   defaultHost: passbook.k8s.local | ||||
|   tls: [] | ||||
|   #  - secretName: chart-example-tls | ||||
|   #    hosts: | ||||
|   #      - passbook.k8s.local | ||||
|  | ||||
| resources: {} | ||||
|   # We usually recommend not to specify default resources and to leave this as a conscious | ||||
|   # choice for the user. This also increases chances charts run on environments with little | ||||
|   # 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 | ||||
| # These settings configure the packaged PostgreSQL and Redis chart. | ||||
| postgresql: | ||||
|   postgresqlDatabase: passbook | ||||
|  | ||||
| nodeSelector: {} | ||||
|  | ||||
| tolerations: [] | ||||
|  | ||||
| affinity: {} | ||||
| redis: | ||||
|   cluster: | ||||
|     enabled: false | ||||
|   master: | ||||
|     persistence: | ||||
|       enabled: false | ||||
|  | ||||
| @ -4,7 +4,7 @@ import os | ||||
| import sys | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.core.settings') | ||||
|     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.root.settings') | ||||
|     try: | ||||
|         from django.core.management import execute_from_command_line | ||||
|     except ImportError as exc: | ||||
|  | ||||
| @ -1,7 +0,0 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| # Check if this file is a symlink, if so, read real base dir | ||||
| BASE_DIR=$(dirname $(readlink -f ${BASH_SOURCE[0]})) | ||||
|  | ||||
| cd $BASE_DIR | ||||
| PYTHONPATH="${BASE_DIR}/vendor/" python3 manage.py $@ | ||||
| @ -1,2 +1,2 @@ | ||||
| """passbook""" | ||||
| __version__ = '0.1.13-beta' | ||||
| __version__ = '0.7.4-beta' | ||||
|  | ||||
| @ -1,2 +0,0 @@ | ||||
| """passbook admin""" | ||||
| __version__ = '0.1.13-beta' | ||||
|  | ||||
| @ -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 | ||||
|  | ||||
| SOURCE_FORM_FIELDS = ['name', 'slug', 'enabled', 'policies'] | ||||
| SOURCE_SERIALIZER_FIELDS = ['pk', 'name', 'slug', 'enabled', 'policies'] | ||||
|  | ||||
| # class SourceForm(forms.Form) | ||||
|  | ||||
| @ -2,6 +2,7 @@ | ||||
|  | ||||
| from django import forms | ||||
|  | ||||
| from passbook.admin.fields import YAMLField | ||||
| from passbook.core.models import User | ||||
|  | ||||
|  | ||||
| @ -11,7 +12,10 @@ class UserForm(forms.ModelForm): | ||||
|     class Meta: | ||||
|  | ||||
|         model = User | ||||
|         fields = ['username', 'name', 'email', 'is_staff', 'is_active'] | ||||
|         fields = ['username', 'name', 'email', 'is_staff', 'is_active', 'attributes'] | ||||
|         widgets = { | ||||
|             'name': forms.TextInput | ||||
|             'name': forms.TextInput, | ||||
|         } | ||||
|         field_classes = { | ||||
|             'attributes': YAMLField, | ||||
|         } | ||||
|  | ||||
| @ -1,2 +0,0 @@ | ||||
| django-rest-framework | ||||
| drf_yasg | ||||
							
								
								
									
										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)}; | ||||
| }); | ||||
|  | ||||
| }); | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	