Compare commits
	
		
			347 Commits
		
	
	
		
			version/0.
			...
			version/0.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 8f42a7f0b4 | |||
| 2c221ea819 | |||
| 93e0441b58 | |||
| 7f1455cb12 | |||
| 59fc223a85 | |||
| 0a6f555c23 | |||
| 6a4233d6fd | |||
| 15fa7e9652 | |||
| f2acc154cd | |||
| d21ec6c9a5 | |||
| 43dd858cd5 | |||
| 34cbf5f702 | |||
| 3c6e94b6a8 | |||
| 1cd149c815 | |||
| 4c6f562805 | |||
| e59c4ec1c7 | |||
| 1169db7530 | |||
| 1453008796 | |||
| 2209b6d603 | |||
| ccbc0384f9 | |||
| a48924c896 | |||
| dc8d8dd2b6 | |||
| afca94ceb8 | |||
| 0b86231a36 | |||
| c0df1f38b8 | |||
| 2b8fed8f4e | |||
| c7322a32a0 | |||
| 64b75cab84 | |||
| f58bc61999 | |||
| fb8ccc0283 | |||
| c38012f147 | |||
| 3676ff21c2 | |||
| 920e705d75 | |||
| de0b137b1e | |||
| d44ac6e2a3 | |||
| 71039a4012 | |||
| 8745ac7932 | |||
| 7f70048423 | |||
| 97dbfc8885 | |||
| 149ea22a93 | |||
| 404ed5406d | |||
| b8656858ec | |||
| 6b0f0e8993 | |||
| aec1ccd88d | |||
| bee5c200b6 | |||
| 9d640efc88 | |||
| f0907841dd | |||
| 2bffc12ef9 | |||
| 2ff9ec6522 | |||
| 43a54f5c54 | |||
| 7bff2734aa | |||
| 84768c0ec6 | |||
| f4499a5459 | |||
| b3aede5bba | |||
| 531ea1c039 | |||
| c2c5ff6912 | |||
| 9cddab8fd5 | |||
| 06d15d8a27 | |||
| b5c711854b | |||
| 4cf6c36f34 | |||
| 75a6f6c875 | |||
| 62abe3f256 | |||
| 9296c41650 | |||
| 7fb48fde6d | |||
| 174472bb45 | |||
| 17575ed921 | |||
| b1b1a27444 | |||
| f97a5eeefb | |||
| 10fd96981e | |||
| 67e3eb549c | |||
| 30a6d1f0b1 | |||
| 3d1fa9f048 | |||
| 1d2be6e68b | |||
| c21e343986 | |||
| ff37ed095c | |||
| 8623a2c3fc | |||
| 23d277eaf1 | |||
| 75ced59451 | |||
| bccf424c5e | |||
| 2f9ae40d20 | |||
| 11e1eec3fb | |||
| 765c5633df | |||
| 6344b1aafb | |||
| ed25801e6e | |||
| 4d0148193f | |||
| 804ae15c2e | |||
| b35a9fad86 | |||
| a4f83bd28a | |||
| 796f83c3d0 | |||
| 2099bbb713 | |||
| 67beba8f78 | |||
| a798412e17 | |||
| 3b2c2d781f | |||
| 98c844f3d6 | |||
| 2645bd0132 | |||
| 2c4fc56b49 | |||
| 0ec1468058 | |||
| 5d1a3043b2 | |||
| b46958d1f9 | |||
| 5daa8d5fe3 | |||
| 31846f1d05 | |||
| 1fac964b8b | |||
| dfa6ed8ac2 | |||
| 66fe10299e | |||
| e0a3ec033f | |||
| 7033ec0ab9 | |||
| 4004579905 | |||
| 9fe9e48a5c | |||
| 595a6c7fe6 | |||
| 11b5860d4a | |||
| 9bdbff4cda | |||
| e0d597eeac | |||
| f576985cc9 | |||
| 22a6aef60b | |||
| ec0a6e7854 | |||
| 6904608e6f | |||
| cb3732cb2b | |||
| 57de6cbafc | |||
| b1dda764a9 | |||
| 5ec2102487 | |||
| 9f8fb7378a | |||
| 98cd646044 | |||
| 0cba1b4c45 | |||
| 53918462b6 | |||
| 8a7e74b523 | |||
| 4dc7065e97 | |||
| 3c93bb9f9f | |||
| 8143fae2d6 | |||
| 3cfe45d3cb | |||
| 8e5c3f2f31 | |||
| 5a3b2fdd49 | |||
| e47b9f0d57 | |||
| 146dd747f1 | |||
| f2ce56063b | |||
| b26f378e4c | |||
| 9072b836c6 | |||
| 2fa57d064e | |||
| 146705c60a | |||
| 5029a99df6 | |||
| e7129d18f6 | |||
| d2bf9f81d6 | |||
| 30acf0660b | |||
| dda41af5c8 | |||
| 9b5b03647b | |||
| 940b3eb943 | |||
| 16eb629b71 | |||
| 755045b226 | |||
| 61478db94e | |||
| f69f959bdb | |||
| 146edb45d4 | |||
| 045a802365 | |||
| c90d8ddcff | |||
| 3ff2ec929f | |||
| a3ef26b7ad | |||
| 19cd1624c1 | |||
| 366ef352c6 | |||
| a9031a6abc | |||
| a1a5223b58 | |||
| c723b0233f | |||
| b369eb28f1 | |||
| 9b8f390e31 | |||
| 11630c9a74 | |||
| c9ac10f6f6 | |||
| 04d613cb28 | |||
| 40866f9ecd | |||
| d8585eb872 | |||
| 15aaeda475 | |||
| 8536ef9e23 | |||
| 35b6bb6b3f | |||
| eaa573c715 | |||
| 660972e303 | |||
| a21012bf0c | |||
| 8dbafa4bda | |||
| 80049413f0 | |||
| 2739442d4a | |||
| c679f0a67c | |||
| d9a952dd03 | |||
| 9a1a0f0aa8 | |||
| 4d6bb60134 | |||
| 80e6d59382 | |||
| 81ac951872 | |||
| f33e553cfd | |||
| 9b0240dc26 | |||
| c327310392 | |||
| 457375287c | |||
| 7e87bfef5b | |||
| a7af5268de | |||
| 6d916029bb | |||
| 81fdcbadad | |||
| ec1e25fe71 | |||
| b5306e4a94 | |||
| 801b8a1e59 | |||
| 3a52059793 | |||
| 10b7d99b37 | |||
| 6be8d0cbb2 | |||
| 5b8e3689ec | |||
| 25a5d8f5da | |||
| 883d439544 | |||
| 1c3b5889e5 | |||
| 87012b65e1 | |||
| 29913773a7 | |||
| 0bc6a4fed4 | |||
| 4645d8353f | |||
| 260c5555fa | |||
| 6f7b917c38 | |||
| 1456ee6d3e | |||
| ae3d3d0295 | |||
| c23ceacd0b | |||
| 5155204283 | |||
| 5509ec9b0f | |||
| d6f9b2e47d | |||
| 67aa4aef11 | |||
| 9e46c8bfec | |||
| 1eaa9b9733 | |||
| ee05834b69 | |||
| fccc8f4959 | |||
| c721620f96 | |||
| c9f73d718e | |||
| bfa58be721 | |||
| 4bb602149e | |||
| 81ab9092fc | |||
| 29d5962c4c | |||
| 5c75339946 | |||
| 4774d9a46c | |||
| dbe16ba4fd | |||
| 6972cf00a0 | |||
| 0445be9712 | |||
| 89dbdd9585 | |||
| da88ce7150 | |||
| 5f50fcfcf5 | |||
| 96be087221 | |||
| a53a269a8c | |||
| 59565a5286 | |||
| ae3c092238 | |||
| e98e5e4e3e | |||
| d50c7ec8d4 | |||
| c0fdf377d1 | |||
| 70c11c8988 | |||
| 67b19becc1 | |||
| ae64024ef4 | |||
| e6571826cb | |||
| c621e61978 | |||
| 3626fa4b98 | |||
| 01b0eb159a | |||
| 63aa48d981 | |||
| 2e0ba05d55 | |||
| b2ac57bb67 | |||
| 4c22e5c2c8 | |||
| 4a7b0ec8a9 | |||
| 330118249e | |||
| 8d4dabde02 | |||
| cf7323c41b | |||
| edd856df7d | |||
| 5e35859db6 | |||
| acabb2df54 | |||
| e6376a05f7 | |||
| 1f45aff7ad | |||
| e1f1f617b6 | |||
| 2690675dca | |||
| 7529b51358 | |||
| c394066d99 | |||
| 9c585032ef | |||
| d408031304 | |||
| c47bc11ec0 | |||
| 1deb094afe | |||
| 501fed1922 | |||
| ad8125ac1c | |||
| b42a551fb2 | |||
| 3256be23df | |||
| f7c0c0146a | |||
| e4baf8c21e | |||
| 364f040b36 | |||
| 2b8c2b2346 | |||
| 5f861189e4 | |||
| 5e11b6687e | |||
| c4b429825d | |||
| eebbae0677 | |||
| 42b30f4507 | |||
| 0e425418df | |||
| 7fe0300b86 | |||
| c012c6be5c | |||
| a5dc193cfd | |||
| 7507ad2620 | |||
| f1291fec8d | |||
| 37aeeea239 | |||
| 0fa1fc86da | |||
| c3034ab9ac | |||
| 76694e037a | |||
| 787db41cc3 | |||
| 74da3df7cd | |||
| a6e435bd70 | |||
| c313b496aa | |||
| a7eaa74191 | |||
| 11ecdc4fcf | |||
| 2f7781b67a | |||
| 296d4f691a | |||
| 64033031b1 | |||
| 9daff7608d | |||
| 0a4af80b9b | |||
| a54adb05c4 | |||
| 43a389e596 | |||
| 2d7e8f1b50 | |||
| cf11f6b121 | |||
| 6dcdf7bcce | |||
| 56d872af15 | |||
| ca663d16fc | |||
| e05c18b19b | |||
| a7b86e46bc | |||
| 84f56674c2 | |||
| 02ab177c6d | |||
| 1232c487e9 | |||
| ef0a2bfbe8 | |||
| 05242a11ad | |||
| 4593ad7bcc | |||
| d7fd5a7fa6 | |||
| 4439378fd4 | |||
| acf65eafdd | |||
| c2ebff55ef | |||
| 99c82676b6 | |||
| 4991e9b825 | |||
| 612f95c3ba | |||
| cd91d5ca15 | |||
| cbbbb5dc08 | |||
| c1640b9411 | |||
| a4842c1f95 | |||
| a4707ddc54 | |||
| fb82d56307 | |||
| 1a1005f80d | |||
| e86cae6cac | |||
| 0b282f45e0 | |||
| 791e88ffc1 | |||
| 7bd3c4bccf | |||
| 722e2e4050 | |||
| c7fc444c95 | |||
| 20ad062814 | |||
| fcb5d36e07 | |||
| 9b131b619f | |||
| 54427f7c68 | |||
| 35eef9c28d | |||
| e88a82553d | |||
| 01a9520140 | |||
| 46667615c3 | |||
| c6721a83a4 | |||
| 46866e8ef0 | |||
| 4a49681127 | |||
| 4c3fced4e9 | |||
| 172347d90f | 
| @ -1,14 +1,15 @@ | |||||||
| [bumpversion] | [bumpversion] | ||||||
| current_version = 0.0.13-alpha | current_version = 0.6.0-beta | ||||||
| tag = True | tag = True | ||||||
| commit = True | commit = True | ||||||
| parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)\-(?P<release>.*) | parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)\-(?P<release>.*) | ||||||
| serialize = {major}.{minor}.{patch}-{release} | serialize = {major}.{minor}.{patch}-{release} | ||||||
| message = bump version: {current_version} -> {new_version} | message = new release: {new_version} | ||||||
| tag_name = version/{new_version} | tag_name = version/{new_version} | ||||||
|  |  | ||||||
| [bumpversion:part:release] | [bumpversion:part:release] | ||||||
| optional_value = stable | optional_value = stable | ||||||
|  | first_value = beta | ||||||
| values =  | values =  | ||||||
| 	alpha | 	alpha | ||||||
| 	beta | 	beta | ||||||
| @ -22,25 +23,5 @@ values = | |||||||
|  |  | ||||||
| [bumpversion:file:passbook/__init__.py] | [bumpversion:file:passbook/__init__.py] | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/api/__init__.py] | [bumpversion:file:docker/nginx.conf] | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/core/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/admin/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/captcha_factor/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/oauth_client/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/ldap/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/lib/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/saml_idp/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/audit/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/oauth_provider/__init__.py] |  | ||||||
|  |  | ||||||
| [bumpversion:file:passbook/otp/__init__.py] |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,12 +1,10 @@ | |||||||
| [run] | [run] | ||||||
| source = passbook | source = passbook | ||||||
| omit = | omit = | ||||||
|     env/ |  | ||||||
|     */wsgi.py |     */wsgi.py | ||||||
|     manage.py |     manage.py | ||||||
|     */migrations/* |     */migrations/* | ||||||
|     */apps.py |     */apps.py | ||||||
|     passbook/management/commands/nexus_upload.py |  | ||||||
|     passbook/management/commands/web.py |     passbook/management/commands/web.py | ||||||
|     passbook/management/commands/worker.py |     passbook/management/commands/worker.py | ||||||
|     docs/ |     docs/ | ||||||
|  | |||||||
| @ -2,3 +2,4 @@ env | |||||||
| helm | helm | ||||||
| passbook-ui | passbook-ui | ||||||
| static | static | ||||||
|  | *.env.yml | ||||||
|  | |||||||
| @ -9,3 +9,6 @@ insert_final_newline = true | |||||||
|  |  | ||||||
| [html] | [html] | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  |  | ||||||
|  | [yaml] | ||||||
|  | indent_size = 2 | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -191,3 +191,4 @@ pip-selfcheck.json | |||||||
| # End of https://www.gitignore.io/api/python,django | # End of https://www.gitignore.io/api/python,django | ||||||
| /static/ | /static/ | ||||||
| local.env.yml | local.env.yml | ||||||
|  | .vscode/ | ||||||
|  | |||||||
							
								
								
									
										198
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							
							
						
						
									
										198
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							| @ -1,139 +1,135 @@ | |||||||
| # Global Variables | # Global Variables | ||||||
| before_script: |  | ||||||
|   - "python3 -m pip install -U virtualenv" |  | ||||||
|   - "virtualenv env" |  | ||||||
|   - "source env/bin/activate" |  | ||||||
|   - "pip3 install -U -r requirements-dev.txt" |  | ||||||
| stages: | stages: | ||||||
|  |   - build-base-image | ||||||
|  |   - build-dev-image | ||||||
|   - test |   - test | ||||||
|   - build |   - build | ||||||
|   - docs |   - package | ||||||
| image: python:3.6 | image: docker.beryju.org/passbook/dev:latest | ||||||
| services: |  | ||||||
|   - postgres:latest |  | ||||||
|  |  | ||||||
| variables: | variables: | ||||||
|   POSTGRES_DB: passbook |   POSTGRES_DB: passbook | ||||||
|   POSTGRES_USER: passbook |   POSTGRES_USER: passbook | ||||||
|   POSTGRES_PASSWORD: 'EK-5jnKfjrGRm<77' |   POSTGRES_PASSWORD: "EK-5jnKfjrGRm<77" | ||||||
|  |  | ||||||
|  | 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 --destination docker.beryju.org/passbook/base:0.6.0-beta | ||||||
|  |   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 --destination docker.beryju.org/passbook/dev:0.6.0-beta | ||||||
|  |   stage: build-dev-image | ||||||
|  |   only: | ||||||
|  |     refs: | ||||||
|  |       - tags | ||||||
|  |       - /^version/.*$/ | ||||||
|  |  | ||||||
| include: |  | ||||||
|   - /allauth/.gitlab-ci.yml |  | ||||||
|  |  | ||||||
| isort: | isort: | ||||||
|   script: |   script: | ||||||
|     - isort -c -sg env |     - isort -c -sg env | ||||||
|   stage: test |   stage: test | ||||||
|  |   services: | ||||||
|  |   - postgres:latest | ||||||
|  |   - redis:latest | ||||||
| migrations: | migrations: | ||||||
|   script: |   script: | ||||||
|     - python manage.py migrate |     - python manage.py migrate | ||||||
|   stage: test |   stage: test | ||||||
| prospector: |   services: | ||||||
|   script: |   - postgres:latest | ||||||
|     - prospector |   - redis:latest | ||||||
|   stage: test | # prospector: | ||||||
| pylint: | #   script: | ||||||
|   script: | #     - prospector | ||||||
|     - pylint passbook | #   stage: test | ||||||
|   stage: test | #   services: | ||||||
|  | #   - postgres:latest | ||||||
|  | #   - redis:latest | ||||||
|  | # pylint: | ||||||
|  | #   script: | ||||||
|  | #     - pylint passbook | ||||||
|  | #   stage: test | ||||||
|  | #   services: | ||||||
|  | #   - postgres:latest | ||||||
|  | #   - redis:latest | ||||||
| coverage: | coverage: | ||||||
|   script: |   script: | ||||||
|     - coverage run manage.py test |     - coverage run manage.py test | ||||||
|     - coverage report |     - coverage report | ||||||
|  |     - coverage html | ||||||
|   stage: test |   stage: test | ||||||
| bandit: |   services: | ||||||
|   script: |   - postgres:latest | ||||||
|     - bandit -r passbook |   - redis:latest | ||||||
|   stage: test |  | ||||||
|  |  | ||||||
| package-docker: | build-passbook-server: | ||||||
|  |   stage: build | ||||||
|   image: |   image: | ||||||
|     name: gcr.io/kaniko-project/executor:debug |     name: gcr.io/kaniko-project/executor:debug | ||||||
|     entrypoint: [""] |     entrypoint: [""] | ||||||
|   before_script: |   before_script: | ||||||
|     - echo "{\"auths\":{\"docker.$NEXUS_URL\":{\"auth\":\"$NEXUS_AUTH\"}}}" > /kaniko/.docker/config.json |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|   script: |   script: | ||||||
|     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/Dockerfile --destination docker.pkg.beryju.org/passbook:latest --destination docker.pkg.beryju.org/passbook:0.0.13-alpha |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/Dockerfile --destination docker.beryju.org/passbook/server:latest --destination docker.beryju.org/passbook/server:0.6.0-beta | ||||||
|   stage: build |  | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
|     - /^version/.*$/ |     - /^version/.*$/ | ||||||
| package-helm: | build-passbook-static: | ||||||
|   stage: build |   stage: build | ||||||
|  |   image: | ||||||
|  |     name: gcr.io/kaniko-project/executor:debug | ||||||
|  |     entrypoint: [""] | ||||||
|  |   before_script: | ||||||
|  |     - echo "{\"auths\":{\"docker.beryju.org\":{\"auth\":\"$DOCKER_AUTH\"}}}" > /kaniko/.docker/config.json | ||||||
|   script: |   script: | ||||||
|     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash |     - /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/static.Dockerfile --destination docker.beryju.org/passbook/static:latest --destination docker.beryju.org/passbook/static:0.6.0-beta | ||||||
|     - helm init --client-only |  | ||||||
|     - helm package helm/passbook |  | ||||||
|     - ./manage.py nexus_upload --method put --url $NEXUS_URL --auth $NEXUS_AUTH --repo helm *.tgz |  | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
|     - /^version/.*$/ |     - /^version/.*$/ | ||||||
| # package-3.5: |   # running collectstatic fully initialises django, hence we need that databases | ||||||
| #   before_script: |   services: | ||||||
| #     - apt update |     - postgres:latest | ||||||
| #     - apt install -y build-essential debhelper devscripts equivs python3 python3-pip |     - redis:latest | ||||||
| #     - cp debian/control-3.5 debian/control |  | ||||||
| #     - mk-build-deps debian/control |  | ||||||
| #     - apt install ./*build-deps*deb -f -y |  | ||||||
| #     - "python3 -m pip install -U virtualenv" |  | ||||||
| #     - "virtualenv env" |  | ||||||
| #     - "source env/bin/activate" |  | ||||||
| #     - "pip3 install -U -r requirements.txt -r requirements-dev.txt" |  | ||||||
| #   image: debian |  | ||||||
| #   script: |  | ||||||
| #     - debuild -us -uc |  | ||||||
| #     - cp ../passbook*.deb . |  | ||||||
| #     - python manage.py nexus_upload |  | ||||||
| #   artifacts: |  | ||||||
| #     paths: |  | ||||||
| #     - passbook-python3.5*deb |  | ||||||
| #     expire_in: 2 days |  | ||||||
| #   stage: build |  | ||||||
| #   only: |  | ||||||
| #   - tags |  | ||||||
| #   - /^debian/.*$/ |  | ||||||
| # package-3.6: |  | ||||||
| #   before_script: |  | ||||||
| #     - apt update |  | ||||||
| #     - apt install -y build-essential debhelper devscripts equivs python3 python3-pip |  | ||||||
| #     - cp debian/control-3.6 debian/control |  | ||||||
| #     - mk-build-deps debian/control |  | ||||||
| #     - apt install ./*build-deps*deb -f -y |  | ||||||
| #     - "python3 -m pip install -U virtualenv" |  | ||||||
| #     - "virtualenv env" |  | ||||||
| #     - "source env/bin/activate" |  | ||||||
| #     - "pip3 install -U -r requirements.txt -r requirements-dev.txt" |  | ||||||
| #   image: debian:buster |  | ||||||
| #   script: |  | ||||||
| #     - debuild -us -uc |  | ||||||
| #     - cp ../passbook*.deb . |  | ||||||
| #     - python manage.py nexus_upload |  | ||||||
| #   artifacts: |  | ||||||
| #     paths: |  | ||||||
| #     - passbook-python3.6*deb |  | ||||||
| #     expire_in: 2 days |  | ||||||
| #   stage: build |  | ||||||
| #   only: |  | ||||||
| #     - tags |  | ||||||
| #     - /^debian/.*$r |  | ||||||
|  |  | ||||||
| # docs: | package-helm: | ||||||
| #   stage: docs |   image: debian:stretch-slim | ||||||
| #   only: |   stage: package | ||||||
| #   - master |   before_script: | ||||||
| #   - tags |     - apt update && apt install -y curl | ||||||
| #   - /^debian/.*$/ |     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||||
| #   environment: |   script: | ||||||
| #     name: docs |     - helm init --client-only | ||||||
| #     url: "https://passbook.beryju.org/docs/" |     - helm dependency update helm/passbook | ||||||
| #   script: |     - helm package helm/passbook | ||||||
| #     - apt update |   artifacts: | ||||||
| #     - apt install -y rsync |     paths: | ||||||
| #     - "mkdir ~/.ssh" |       - passbook-*.tgz | ||||||
| #     - "cp .gitlab/known_hosts ~/.ssh/" |     expire_in: 1 week | ||||||
| #     - "pip3 install -U -r requirements-docs.txt" |   only: | ||||||
| #     - "eval $(ssh-agent -s)" |     - tags | ||||||
| #     - "echo \"${CI_SSH_PRIVATE}\" | ssh-add -" |     - /^version/.*$/ | ||||||
| #     - mkdocs build |  | ||||||
| #     - 'rsync -avh --delete web/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/"' |  | ||||||
| #     - 'rsync -avh --delete site/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/docs/"' |  | ||||||
|  | |||||||
| @ -3,7 +3,6 @@ test-warnings: true | |||||||
| doc-warnings: false | doc-warnings: false | ||||||
|  |  | ||||||
| ignore-paths: | ignore-paths: | ||||||
|   - env |  | ||||||
|   - migrations |   - migrations | ||||||
|   - docs |   - docs | ||||||
|   - node_modules |   - node_modules | ||||||
|  | |||||||
| @ -2,9 +2,10 @@ | |||||||
|  |  | ||||||
| disable=redefined-outer-name,arguments-differ,no-self-use,cyclic-import,fixme,locally-disabled,unpacking-non-sequence,too-many-ancestors,too-many-branches,too-few-public-methods | disable=redefined-outer-name,arguments-differ,no-self-use,cyclic-import,fixme,locally-disabled,unpacking-non-sequence,too-many-ancestors,too-many-branches,too-few-public-methods | ||||||
| load-plugins=pylint_django,pylint.extensions.bad_builtin | load-plugins=pylint_django,pylint.extensions.bad_builtin | ||||||
| #,pylint.extensions.docparams |  | ||||||
| extension-pkg-whitelist=lxml | extension-pkg-whitelist=lxml | ||||||
| const-rgx=[a-zA-Z0-9_]{1,40}$ | const-rgx=[a-zA-Z0-9_]{1,40}$ | ||||||
|  | ignored-modules=django-otp | ||||||
|  | jobs=4 | ||||||
|  |  | ||||||
| [SIMILARITIES] | [SIMILARITIES] | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										114
									
								
								.vscode/.ropeproject/config.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										114
									
								
								.vscode/.ropeproject/config.py
									
									
									
									
										vendored
									
									
								
							| @ -1,114 +0,0 @@ | |||||||
| # The default ``config.py`` |  | ||||||
| # flake8: noqa |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def set_prefs(prefs): |  | ||||||
|     """This function is called before opening the project""" |  | ||||||
|  |  | ||||||
|     # Specify which files and folders to ignore in the project. |  | ||||||
|     # Changes to ignored resources are not added to the history and |  | ||||||
|     # VCSs.  Also they are not returned in `Project.get_files()`. |  | ||||||
|     # Note that ``?`` and ``*`` match all characters but slashes. |  | ||||||
|     # '*.pyc': matches 'test.pyc' and 'pkg/test.pyc' |  | ||||||
|     # 'mod*.pyc': matches 'test/mod1.pyc' but not 'mod/1.pyc' |  | ||||||
|     # '.svn': matches 'pkg/.svn' and all of its children |  | ||||||
|     # 'build/*.o': matches 'build/lib.o' but not 'build/sub/lib.o' |  | ||||||
|     # 'build//*.o': matches 'build/lib.o' and 'build/sub/lib.o' |  | ||||||
|     prefs['ignored_resources'] = ['*.pyc', '*~', '.ropeproject', |  | ||||||
|                                   '.hg', '.svn', '_svn', '.git', '.tox'] |  | ||||||
|  |  | ||||||
|     # Specifies which files should be considered python files.  It is |  | ||||||
|     # useful when you have scripts inside your project.  Only files |  | ||||||
|     # ending with ``.py`` are considered to be python files by |  | ||||||
|     # default. |  | ||||||
|     # prefs['python_files'] = ['*.py'] |  | ||||||
|  |  | ||||||
|     # Custom source folders:  By default rope searches the project |  | ||||||
|     # for finding source folders (folders that should be searched |  | ||||||
|     # for finding modules).  You can add paths to that list.  Note |  | ||||||
|     # that rope guesses project source folders correctly most of the |  | ||||||
|     # time; use this if you have any problems. |  | ||||||
|     # The folders should be relative to project root and use '/' for |  | ||||||
|     # separating folders regardless of the platform rope is running on. |  | ||||||
|     # 'src/my_source_folder' for instance. |  | ||||||
|     # prefs.add('source_folders', 'src') |  | ||||||
|  |  | ||||||
|     # You can extend python path for looking up modules |  | ||||||
|     # prefs.add('python_path', '~/python/') |  | ||||||
|  |  | ||||||
|     # Should rope save object information or not. |  | ||||||
|     prefs['save_objectdb'] = True |  | ||||||
|     prefs['compress_objectdb'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope analyzes each module when it is being saved. |  | ||||||
|     prefs['automatic_soa'] = True |  | ||||||
|     # The depth of calls to follow in static object analysis |  | ||||||
|     prefs['soa_followed_calls'] = 0 |  | ||||||
|  |  | ||||||
|     # If `False` when running modules or unit tests "dynamic object |  | ||||||
|     # analysis" is turned off.  This makes them much faster. |  | ||||||
|     prefs['perform_doa'] = True |  | ||||||
|  |  | ||||||
|     # Rope can check the validity of its object DB when running. |  | ||||||
|     prefs['validate_objectdb'] = True |  | ||||||
|  |  | ||||||
|     # How many undos to hold? |  | ||||||
|     prefs['max_history_items'] = 32 |  | ||||||
|  |  | ||||||
|     # Shows whether to save history across sessions. |  | ||||||
|     prefs['save_history'] = True |  | ||||||
|     prefs['compress_history'] = False |  | ||||||
|  |  | ||||||
|     # Set the number spaces used for indenting.  According to |  | ||||||
|     # :PEP:`8`, it is best to use 4 spaces.  Since most of rope's |  | ||||||
|     # unit-tests use 4 spaces it is more reliable, too. |  | ||||||
|     prefs['indent_size'] = 4 |  | ||||||
|  |  | ||||||
|     # Builtin and c-extension modules that are allowed to be imported |  | ||||||
|     # and inspected by rope. |  | ||||||
|     prefs['extension_modules'] = [] |  | ||||||
|  |  | ||||||
|     # Add all standard c-extensions to extension_modules list. |  | ||||||
|     prefs['import_dynload_stdmods'] = True |  | ||||||
|  |  | ||||||
|     # If `True` modules with syntax errors are considered to be empty. |  | ||||||
|     # The default value is `False`; When `False` syntax errors raise |  | ||||||
|     # `rope.base.exceptions.ModuleSyntaxError` exception. |  | ||||||
|     prefs['ignore_syntax_errors'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope ignores unresolvable imports.  Otherwise, they |  | ||||||
|     # appear in the importing namespace. |  | ||||||
|     prefs['ignore_bad_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will insert new module imports as |  | ||||||
|     # `from <package> import <module>` by default. |  | ||||||
|     prefs['prefer_module_from_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will transform a comma list of imports into |  | ||||||
|     # multiple separate import statements when organizing |  | ||||||
|     # imports. |  | ||||||
|     prefs['split_imports'] = False |  | ||||||
|  |  | ||||||
|     # If `True`, rope will remove all top-level import statements and |  | ||||||
|     # reinsert them at the top of the module when making changes. |  | ||||||
|     prefs['pull_imports_to_top'] = True |  | ||||||
|  |  | ||||||
|     # If `True`, rope will sort imports alphabetically by module name instead |  | ||||||
|     # of alphabetically by import statement, with from imports after normal |  | ||||||
|     # imports. |  | ||||||
|     prefs['sort_imports_alphabetically'] = False |  | ||||||
|  |  | ||||||
|     # Location of implementation of |  | ||||||
|     # rope.base.oi.type_hinting.interfaces.ITypeHintingFactory In general |  | ||||||
|     # case, you don't have to change this value, unless you're an rope expert. |  | ||||||
|     # Change this value to inject you own implementations of interfaces |  | ||||||
|     # listed in module rope.base.oi.type_hinting.providers.interfaces |  | ||||||
|     # For example, you can add you own providers for Django Models, or disable |  | ||||||
|     # the search type-hinting in a class hierarchy, etc. |  | ||||||
|     prefs['type_hinting_factory'] = ( |  | ||||||
|         'rope.base.oi.type_hinting.factory.default_type_hinting_factory') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def project_opened(project): |  | ||||||
|     """This function is called after opening the project""" |  | ||||||
|     # Do whatever you like here! |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								.vscode/.ropeproject/objectdb
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								.vscode/.ropeproject/objectdb
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										11
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @ -1,11 +0,0 @@ | |||||||
| { |  | ||||||
|   "python.pythonPath": "env/bin/python", |  | ||||||
|   "editor.tabSize": 4, |  | ||||||
|   "[html]": { |  | ||||||
|     "editor.tabSize": 2 |  | ||||||
|   }, |  | ||||||
|   "cSpell.words": [ |  | ||||||
|     "SAML", |  | ||||||
|     "passbook" |  | ||||||
|   ] |  | ||||||
| } |  | ||||||
							
								
								
									
										27
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -1,28 +1,9 @@ | |||||||
| FROM python:3.6-slim-stretch as build | FROM docker.beryju.org/passbook/base:latest | ||||||
|  |  | ||||||
| COPY ./passbook/ /app/passbook | COPY ./passbook/ /app/passbook | ||||||
| COPY ./manage.py /app/ | COPY ./manage.py /app/ | ||||||
| COPY ./requirements.txt /app/ | COPY ./docker/uwsgi.ini /app/ | ||||||
|  |  | ||||||
| WORKDIR /app/ |  | ||||||
|  |  | ||||||
| RUN mkdir /app/static/ && \ |  | ||||||
|     pip install -r requirements.txt && \ |  | ||||||
|     pip install psycopg2 && \ |  | ||||||
|     ./manage.py collectstatic --no-input |  | ||||||
|  |  | ||||||
| FROM python:3.6-slim-stretch |  | ||||||
|  |  | ||||||
| COPY ./passbook/ /app/passbook |  | ||||||
| COPY ./manage.py /app/ |  | ||||||
| COPY ./requirements.txt /app/ |  | ||||||
| COPY --from=build /app/static /app/static/ |  | ||||||
|  |  | ||||||
| WORKDIR /app/ |  | ||||||
|  |  | ||||||
| RUN pip install -r requirements.txt && \ |  | ||||||
|     pip install psycopg2 && \ |  | ||||||
|     adduser --system --home /app/ passbook && \ |  | ||||||
|     chown -R passbook /app/ |  | ||||||
|  |  | ||||||
| USER passbook | USER passbook | ||||||
|  |  | ||||||
|  | WORKDIR /app/ | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @ -1,6 +1,6 @@ | |||||||
| MIT License | MIT License | ||||||
|  |  | ||||||
| Copyright (c) 2018 BeryJu.org | Copyright (c) 2019 BeryJu.org | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
| of this software and associated documentation files (the "Software"), to deal | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | |||||||
							
								
								
									
										54
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,54 @@ | |||||||
|  | [[source]] | ||||||
|  | name = "pypi" | ||||||
|  | url = "https://pypi.org/simple" | ||||||
|  | verify_ssl = true | ||||||
|  |  | ||||||
|  | [packages] | ||||||
|  | celery = "*" | ||||||
|  | cherrypy = "*" | ||||||
|  | defusedxml = "*" | ||||||
|  | django = "*" | ||||||
|  | django-cors-middleware = "*" | ||||||
|  | django-filters = "*" | ||||||
|  | django-ipware = "*" | ||||||
|  | django-model-utils = "*" | ||||||
|  | django-oauth-toolkit = "*" | ||||||
|  | django-oidc-provider = "*" | ||||||
|  | django-otp = "*" | ||||||
|  | django-recaptcha = "*" | ||||||
|  | django-redis = "*" | ||||||
|  | django-rest-framework = "*" | ||||||
|  | djangorestframework = "==3.9.4" | ||||||
|  | drf-yasg = "*" | ||||||
|  | ldap3 = "*" | ||||||
|  | lxml = "*" | ||||||
|  | markdown = "*" | ||||||
|  | oauthlib = "*" | ||||||
|  | packaging = "*" | ||||||
|  | psycopg2-binary = "*" | ||||||
|  | pycryptodome = "*" | ||||||
|  | pyyaml = "*" | ||||||
|  | qrcode = "*" | ||||||
|  | requests-oauthlib = "*" | ||||||
|  | sentry-sdk = "*" | ||||||
|  | service_identity = "*" | ||||||
|  | signxml = "*" | ||||||
|  | urllib3 = {extras = ["secure"],version = "*"} | ||||||
|  | structlog = "*" | ||||||
|  |  | ||||||
|  | [requires] | ||||||
|  | python_version = "3.7" | ||||||
|  |  | ||||||
|  | [dev-packages] | ||||||
|  | astroid = "==2.2.5" | ||||||
|  | coverage = "*" | ||||||
|  | isort = "*" | ||||||
|  | pylint = "==2.3.1" | ||||||
|  | pylint-django = "==2.0.10" | ||||||
|  | prospector = "==1.1.7" | ||||||
|  | django-debug-toolbar = "*" | ||||||
|  | bumpversion = "*" | ||||||
|  | unittest-xml-reporting = "*" | ||||||
|  | autopep8 = "*" | ||||||
|  | bandit = "*" | ||||||
|  | colorama = "*" | ||||||
							
								
								
									
										1086
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										1086
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										9
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # passbook | ||||||
|  |  | ||||||
|  | ## Quick instance | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | docker-compose pull | ||||||
|  | docker-compose up -d | ||||||
|  | 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', |  | ||||||
|     ], |  | ||||||
| ) |  | ||||||
							
								
								
									
										19
									
								
								base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | FROM python:3.7-slim-stretch | ||||||
|  |  | ||||||
|  | COPY ./Pipfile /app/ | ||||||
|  | COPY ./Pipfile.lock /app/ | ||||||
|  |  | ||||||
|  | WORKDIR /app/ | ||||||
|  |  | ||||||
|  | RUN apt-get update && \ | ||||||
|  |     apt-get install -y --no-install-recommends build-essential && \ | ||||||
|  |     pip install pipenv uwsgi --no-cache-dir && \ | ||||||
|  |     apt-get remove -y --purge build-essential && \ | ||||||
|  |     apt-get autoremove -y --purge && \ | ||||||
|  |     rm -rf /var/lib/apt/lists/* | ||||||
|  |  | ||||||
|  | RUN pipenv lock -r > requirements.txt && \ | ||||||
|  |     pipenv --rm && \ | ||||||
|  |     pip install -r requirements.txt  --no-cache-dir && \ | ||||||
|  |     adduser --system --no-create-home passbook && \ | ||||||
|  |     chown -R passbook /app | ||||||
							
								
								
									
										5
									
								
								dev.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								dev.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | FROM docker.beryju.org/passbook/base:latest | ||||||
|  |  | ||||||
|  | RUN pipenv lock --dev -r > requirements-dev.txt && \ | ||||||
|  |     pipenv --rm && \ | ||||||
|  |     pip install -r /app/requirements-dev.txt  --no-cache-dir | ||||||
							
								
								
									
										97
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,97 @@ | |||||||
|  | --- | ||||||
|  | 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 | ||||||
|  |   database-migrate: | ||||||
|  |     build: | ||||||
|  |       context: . | ||||||
|  |     image: docker.beryju.org/passbook/server:${TAG:-test} | ||||||
|  |     command: | ||||||
|  |       - ./manage.py | ||||||
|  |       - migrate | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     restart: 'no' | ||||||
|  |     environment: | ||||||
|  |       - PASSBOOK_REDIS__HOST=redis | ||||||
|  |       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||||
|  |       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |   server: | ||||||
|  |     build: | ||||||
|  |       context: . | ||||||
|  |     image: docker.beryju.org/passbook/server:${TAG:-test} | ||||||
|  |     command: | ||||||
|  |       - uwsgi | ||||||
|  |       - uwsgi.ini | ||||||
|  |     environment: | ||||||
|  |       - PASSBOOK_REDIS__HOST=redis | ||||||
|  |       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||||
|  |       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |     ports: | ||||||
|  |       - 8000 | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.port=8000 | ||||||
|  |       - traefik.docker.network=internal | ||||||
|  |       - traefik.frontend.rule=PathPrefix:/ | ||||||
|  |   worker: | ||||||
|  |     image: docker.beryju.org/passbook/server:${TAG:-test} | ||||||
|  |     command: | ||||||
|  |       - ./manage.py | ||||||
|  |       - worker | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.enable=false | ||||||
|  |     environment: | ||||||
|  |       - PASSBOOK_REDIS__HOST=redis | ||||||
|  |       - PASSBOOK_POSTGRESQL__HOST=postgresql | ||||||
|  |       - PASSBOOK_POSTGRESQL__PASSWORD=${PG_PASS:-thisisnotagoodpassword} | ||||||
|  |   static: | ||||||
|  |     build: | ||||||
|  |       context: . | ||||||
|  |       dockerfile: static.Dockerfile | ||||||
|  |     image: docker.beryju.org/passbook/static:${TAG:-test} | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |     labels: | ||||||
|  |       - traefik.frontend.rule=PathPrefix:/static, /robots.txt | ||||||
|  |       - traefik.port=80 | ||||||
|  |       - traefik.docker.network=internal | ||||||
|  |   traefik: | ||||||
|  |     image: traefik:1.7 | ||||||
|  |     command: --api --docker | ||||||
|  |     volumes: | ||||||
|  |       - /var/run/docker.sock:/var/run/docker.sock:ro | ||||||
|  |     ports: | ||||||
|  |       - "0.0.0.0:80:80" | ||||||
|  |       - "0.0.0.0:443:443" | ||||||
|  |       - "0.0.0.0:8080:8080" | ||||||
|  |     networks: | ||||||
|  |       - internal | ||||||
|  |  | ||||||
|  | volumes: | ||||||
|  |   database: | ||||||
|  |     driver: local | ||||||
|  |  | ||||||
|  | networks: | ||||||
|  |   internal: {} | ||||||
							
								
								
									
										66
									
								
								docker/nginx.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								docker/nginx.conf
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | |||||||
|  | user  nginx; | ||||||
|  | worker_processes  1; | ||||||
|  |  | ||||||
|  | error_log  stderr warn; | ||||||
|  | pid        /var/run/nginx.pid; | ||||||
|  |  | ||||||
|  | events { | ||||||
|  |     worker_connections  1024; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | http { | ||||||
|  |     include       /etc/nginx/mime.types; | ||||||
|  |     default_type  application/octet-stream; | ||||||
|  |  | ||||||
|  |     log_format json_combined escape=json | ||||||
|  |         '{' | ||||||
|  |             '"time_local":"$time_local",' | ||||||
|  |             '"remote_addr":"$remote_addr",' | ||||||
|  |             '"remote_user":"$remote_user",' | ||||||
|  |             '"request":"$request",' | ||||||
|  |             '"status": "$status",' | ||||||
|  |             '"body_bytes_sent":"$body_bytes_sent",' | ||||||
|  |             '"request_time":"$request_time",' | ||||||
|  |             '"http_referrer":"$http_referer",' | ||||||
|  |             '"http_user_agent":"$http_user_agent"' | ||||||
|  |         '}'; | ||||||
|  |  | ||||||
|  |     access_log /dev/stdout json_combined; | ||||||
|  |  | ||||||
|  |     sendfile        on; | ||||||
|  |     tcp_nopush     on; | ||||||
|  |  | ||||||
|  |     keepalive_timeout  65; | ||||||
|  |  | ||||||
|  |     server { | ||||||
|  |  | ||||||
|  |         server_name _; | ||||||
|  |  | ||||||
|  |         gzip on; | ||||||
|  |         gzip_types application/javascript image/* text/css; | ||||||
|  |         gunzip on; | ||||||
|  |         add_header X-passbook-Version 0.6.0-beta; | ||||||
|  |         add_header Vary X-passbook-Version; | ||||||
|  |         root /data/; | ||||||
|  |  | ||||||
|  |         location /_/healthz { | ||||||
|  |             return 204; | ||||||
|  |         } | ||||||
|  |         location ~* \.(jpg|jpeg|png|gif|ico)$ { | ||||||
|  |             expires 30d; | ||||||
|  |         } | ||||||
|  |         location ~* \.(css|js)$ { | ||||||
|  |             expires 7d; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     server { | ||||||
|  |  | ||||||
|  |         listen 8080; | ||||||
|  |  | ||||||
|  |         location = /stub_status { | ||||||
|  |             stub_status; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										10
									
								
								docker/uwsgi.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								docker/uwsgi.ini
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | [uwsgi] | ||||||
|  | http = 0.0.0.0:8000 | ||||||
|  | chdir = /app | ||||||
|  | wsgi-file = passbook/root/wsgi.py | ||||||
|  | processes = 4 | ||||||
|  | master = true | ||||||
|  | threads = 2 | ||||||
|  | enable-threads = true | ||||||
|  | uid = passbook | ||||||
|  | gid = passbook | ||||||
							
								
								
									
										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.3.10 | ||||||
|  | - name: redis | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 9.2.1 | ||||||
|  | digest: sha256:bdde250e1401dccdd5161e39c807f9e88b05e3e8e72e74df767a1bbb5fc39a4a | ||||||
|  | generated: "2019-10-01T10:46:06.542706+02:00" | ||||||
| @ -1,6 +1,6 @@ | |||||||
| apiVersion: v1 | apiVersion: v1 | ||||||
| appVersion: "0.0.13-alpha" | appVersion: "0.6.0-beta" | ||||||
| description: A Helm chart for passbook. | description: A Helm chart for passbook. | ||||||
| name: passbook | name: passbook | ||||||
| version: "0.0.13-alpha" | version: "0.6.0-beta" | ||||||
| icon: https://passbook.beryju.org/images/logo.png | icon: https://git.beryju.org/uploads/-/system/project/avatar/108/logo.png | ||||||
|  | |||||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								helm/passbook/charts/postgresql-4.2.2.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								helm/passbook/charts/postgresql-4.2.2.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								helm/passbook/charts/redis-9.2.1.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								helm/passbook/charts/redis-9.2.1.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @ -1,9 +1,9 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis |  | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |  | ||||||
|   version: 5.1.0 |  | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|   version: 3.10.1 |   version: 4.2.2 | ||||||
| digest: sha256:04bd136761f070e94a2ff32ff48ff87f5e07fbd451e5fd7f65551e3bd4680e5e | - name: redis | ||||||
| generated: 2019-02-08T12:08:49.090666+01:00 |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 9.2.1 | ||||||
|  | digest: sha256:8782e974a1094eaeecf1d68f093ca4fb84977217b2bd38b09790a05ec289aec2 | ||||||
|  | generated: "2019-10-02T21:03:25.90491153Z" | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis |  | ||||||
|   version: 5.1.0 |  | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |  | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   version: 3.10.1 |   version: 4.2.2 | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | - name: redis | ||||||
|  |   version: 9.2.1 | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | |||||||
| @ -4,40 +4,16 @@ metadata: | |||||||
|   name: {{ include "passbook.fullname" . }}-config |   name: {{ include "passbook.fullname" . }}-config | ||||||
| data: | data: | ||||||
|   config.yml: | |   config.yml: | | ||||||
|     # Env for Docker images |     postgresql: | ||||||
|     databases: |       host: "{{ .Release.Name }}-postgresql" | ||||||
|       default: |       name: "{{ .Values.postgresql.postgresqlDatabase }}" | ||||||
|         engine: django.db.backends.postgresql |  | ||||||
|         name: {{ .Values.postgresql.postgresqlDatabase }} |  | ||||||
|       user: postgres |       user: postgres | ||||||
|         password: {{ .Values.postgresql.postgresqlPassword }} |     redis: | ||||||
|         host: {{ .Release.Name }}-postgresql |       host: "{{ .Release.Name }}-redis-master" | ||||||
|         port: '' |       cache_db: 0 | ||||||
|     log: |       message_queue_db: 1 | ||||||
|       level: | 
 | ||||||
|         console: DEBUG |     # Error reporting, sends stacktrace to sentry.beryju.org | ||||||
|         file: DEBUG |  | ||||||
|       file: /dev/null |  | ||||||
|       syslog: |  | ||||||
|         host: 127.0.0.1 |  | ||||||
|         port: 514 |  | ||||||
|     email: |  | ||||||
|       host: localhost |  | ||||||
|       port: 25 |  | ||||||
|       user: '' |  | ||||||
|       password: '' |  | ||||||
|       use_tls: false |  | ||||||
|       use_ssl: false |  | ||||||
|       from: passbook <passbook@domain.tld> |  | ||||||
|     web: |  | ||||||
|       listen: 0.0.0.0 |  | ||||||
|       port: 8000 |  | ||||||
|       threads: 30 |  | ||||||
|     debug: false |  | ||||||
|     secure_proxy_header: |  | ||||||
|       HTTP_X_FORWARDED_PROTO: https |  | ||||||
|     redis: ":{{ .Values.redis.password }}@{{ .Release.Name }}-redis-master" |  | ||||||
|     # Error reporting, sends stacktrace to sentry.services.beryju.org |  | ||||||
|     error_report_enabled: {{ .Values.config.error_reporting }} |     error_report_enabled: {{ .Values.config.error_reporting }} | ||||||
| 
 | 
 | ||||||
|     {{- if .Values.config.secret_key }} |     {{- if .Values.config.secret_key }} | ||||||
| @ -46,10 +22,12 @@ data: | |||||||
|     secret_key: {{ randAlphaNum 50 }} |     secret_key: {{ randAlphaNum 50 }} | ||||||
|     {{- end }} |     {{- end }} | ||||||
| 
 | 
 | ||||||
|  |     primary_domain: {{ .Values.primary_domain }} | ||||||
|     domains: |     domains: | ||||||
|       {{- range .Values.ingress.hosts }} |       {{- range .Values.ingress.hosts }} | ||||||
|       - {{ . | quote }} |       - {{ . | quote }} | ||||||
|       {{- end }} |       {{- end }} | ||||||
|  |       - kubernetes-healthcheck-host | ||||||
| 
 | 
 | ||||||
|     passbook: |     passbook: | ||||||
|       sign_up: |       sign_up: | ||||||
| @ -112,26 +90,9 @@ data: | |||||||
|       # create_users: true |       # create_users: true | ||||||
|       # # Reset LDAP password when user reset their password |       # # Reset LDAP password when user reset their password | ||||||
|       # reset_password: true |       # 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: |     saml_idp: | ||||||
|       signing: true |       signing: true | ||||||
|       autosubmit: false |       autosubmit: false | ||||||
|       issuer: passbook |       issuer: passbook | ||||||
|       assertion_valid_for: 86400 |       assertion_valid_for: 86400 | ||||||
|       # List of python packages with provider types to load. |       # List of python packages with provider types to load. | ||||||
|       types: |  | ||||||
|         - passbook.saml_idp.processors.generic |  | ||||||
|         - passbook.saml_idp.processors.gitlab |  | ||||||
|         - passbook.saml_idp.processors.nextcloud |  | ||||||
|         - passbook.saml_idp.processors.salesforce |  | ||||||
|         - passbook.saml_idp.processors.shibboleth |  | ||||||
|         - passbook.saml_idp.processors.wordpress_orange |  | ||||||
| @ -1,6 +1,5 @@ | |||||||
| {{- if .Values.ingress.enabled -}} | {{- if .Values.ingress.enabled -}} | ||||||
| {{- $fullName := include "passbook.fullname" . -}} | {{- $fullName := include "passbook.fullname" . -}} | ||||||
| {{- $ingressPath := .Values.ingress.path -}} |  | ||||||
| apiVersion: extensions/v1beta1 | apiVersion: extensions/v1beta1 | ||||||
| kind: Ingress | kind: Ingress | ||||||
| metadata: | metadata: | ||||||
| @ -30,9 +29,17 @@ spec: | |||||||
|     - host: {{ . | quote }} |     - host: {{ . | quote }} | ||||||
|       http: |       http: | ||||||
|         paths: |         paths: | ||||||
|           - path: {{ $ingressPath }} |           - path: / | ||||||
|             backend: |             backend: | ||||||
|               serviceName: {{ $fullName }} |               serviceName: {{ $fullName }}-web | ||||||
|  |               servicePort: http | ||||||
|  |           - path: /static/ | ||||||
|  |             backend: | ||||||
|  |               serviceName: {{ $fullName }}-static | ||||||
|  |               servicePort: http | ||||||
|  |           - path: /robots.txt | ||||||
|  |             backend: | ||||||
|  |               serviceName: {{ $fullName }}-static | ||||||
|               servicePort: http |               servicePort: http | ||||||
|   {{- end }} |   {{- end }} | ||||||
| {{- end }} | {{- end }} | ||||||
| @ -1,66 +0,0 @@ | |||||||
| apiVersion: apps/v1beta2 |  | ||||||
| kind: Deployment |  | ||||||
| metadata: |  | ||||||
|   name: {{ include "passbook.fullname" . }}-web |  | ||||||
|   labels: |  | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |  | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     app.kubernetes.io/managed-by: {{ .Release.Service }} |  | ||||||
| spec: |  | ||||||
|   replicas: {{ .Values.replicaCount }} |  | ||||||
|   selector: |  | ||||||
|     matchLabels: |  | ||||||
|       app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|       app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|   template: |  | ||||||
|     metadata: |  | ||||||
|       labels: |  | ||||||
|         app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|         app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     spec: |  | ||||||
|       volumes: |  | ||||||
|         - name: config-volume |  | ||||||
|           configMap: |  | ||||||
|             name: {{ include "passbook.fullname" . }}-config |  | ||||||
|       containers: |  | ||||||
|         - name: {{ .Chart.Name }} |  | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |  | ||||||
|           imagePullPolicy: IfNotPresent |  | ||||||
|           command: ["/bin/sh","-c"] |  | ||||||
|           args: ["./manage.py migrate && ./manage.py web"] |  | ||||||
|           ports: |  | ||||||
|             - name: http |  | ||||||
|               containerPort: 8000 |  | ||||||
|               protocol: TCP |  | ||||||
|           volumeMounts: |  | ||||||
|             - mountPath: /etc/passbook |  | ||||||
|               name: config-volume |  | ||||||
|           livenessProbe: |  | ||||||
|             httpGet: |  | ||||||
|               path: / |  | ||||||
|               port: http |  | ||||||
|               httpHeaders: |  | ||||||
|                 - name: Host |  | ||||||
|                   value: kubernetes-healthcheck-host |  | ||||||
|           readinessProbe: |  | ||||||
|             httpGet: |  | ||||||
|               path: / |  | ||||||
|               port: http |  | ||||||
|               httpHeaders: |  | ||||||
|                 - name: Host |  | ||||||
|                   value: kubernetes-healthcheck-host |  | ||||||
|           resources: |  | ||||||
| {{ toYaml .Values.resources | indent 12 }} |  | ||||||
|     {{- with .Values.nodeSelector }} |  | ||||||
|       nodeSelector: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.affinity }} |  | ||||||
|       affinity: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.tolerations }} |  | ||||||
|       tolerations: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
| @ -1,51 +0,0 @@ | |||||||
| apiVersion: apps/v1beta2 |  | ||||||
| kind: Deployment |  | ||||||
| metadata: |  | ||||||
|   name: {{ include "passbook.fullname" . }}-worker |  | ||||||
|   labels: |  | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |  | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     app.kubernetes.io/managed-by: {{ .Release.Service }} |  | ||||||
| spec: |  | ||||||
|   replicas: {{ .Values.replicaCount }} |  | ||||||
|   selector: |  | ||||||
|     matchLabels: |  | ||||||
|       app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|       app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|   template: |  | ||||||
|     metadata: |  | ||||||
|       labels: |  | ||||||
|         app.kubernetes.io/name: {{ include "passbook.name" . }} |  | ||||||
|         app.kubernetes.io/instance: {{ .Release.Name }} |  | ||||||
|     spec: |  | ||||||
|       volumes: |  | ||||||
|         - name: config-volume |  | ||||||
|           configMap: |  | ||||||
|             name: {{ include "passbook.fullname" . }}-config |  | ||||||
|       containers: |  | ||||||
|         - name: {{ .Chart.Name }} |  | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |  | ||||||
|           imagePullPolicy: IfNotPresent |  | ||||||
|           command: ["./manage.py", "worker"] |  | ||||||
|           ports: |  | ||||||
|             - name: http |  | ||||||
|               containerPort: 8000 |  | ||||||
|               protocol: TCP |  | ||||||
|           volumeMounts: |  | ||||||
|             - mountPath: /etc/passbook |  | ||||||
|               name: config-volume |  | ||||||
|           resources: |  | ||||||
| {{ toYaml .Values.resources | indent 12 }} |  | ||||||
|     {{- with .Values.nodeSelector }} |  | ||||||
|       nodeSelector: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.affinity }} |  | ||||||
|       affinity: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
|     {{- with .Values.tolerations }} |  | ||||||
|       tolerations: |  | ||||||
| {{ toYaml . | indent 8 }} |  | ||||||
|     {{- end }} |  | ||||||
							
								
								
									
										55
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								helm/passbook/templates/static-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | |||||||
|  | apiVersion: apps/v1beta2 | ||||||
|  | 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 | ||||||
|  |       annotations: | ||||||
|  |         prometheus.io/scrape: "true" | ||||||
|  |         prometheus.io/port: '9113' | ||||||
|  |         field.cattle.io/workloadMetrics: '[{"path":"/metrics","port":9113,"schema":"HTTP"}]' | ||||||
|  |     spec: | ||||||
|  |       containers: | ||||||
|  |         - name: {{ .Chart.Name }}-static | ||||||
|  |           image: "docker.beryju.org/passbook/static:{{ .Values.image.tag }}" | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
|  |           ports: | ||||||
|  |             - name: http | ||||||
|  |               containerPort: 80 | ||||||
|  |               protocol: TCP | ||||||
|  |           livenessProbe: | ||||||
|  |             initialDelaySeconds: 10 | ||||||
|  |             timeoutSeconds: 5 | ||||||
|  |             httpGet: | ||||||
|  |               path: /_/healthz | ||||||
|  |               port: http | ||||||
|  |           readinessProbe: | ||||||
|  |             initialDelaySeconds: 10 | ||||||
|  |             timeoutSeconds: 5 | ||||||
|  |             httpGet: | ||||||
|  |               path: /_/healthz | ||||||
|  |               port: http | ||||||
|  |           resources: | ||||||
|  |             requests: | ||||||
|  |               cpu: 10m | ||||||
|  |               memory: 10M | ||||||
|  |             limits: | ||||||
|  |               cpu: 20m | ||||||
|  |               memory: 20M | ||||||
|  |         - name: {{ .Chart.Name }}-static-prometheus | ||||||
|  |           image: nginx/nginx-prometheus-exporter:0.4.1 | ||||||
|  |           imagePullPolicy: IfNotPresent | ||||||
							
								
								
									
										21
									
								
								helm/passbook/templates/static-service.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								helm/passbook/templates/static-service.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | apiVersion: v1 | ||||||
|  | kind: Service | ||||||
|  | metadata: | ||||||
|  |   name: {{ include "passbook.fullname" . }}-static | ||||||
|  |   labels: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     app.kubernetes.io/managed-by: {{ .Release.Service }} | ||||||
|  |     k8s.passbook.io/component: static | ||||||
|  | spec: | ||||||
|  |   type: ClusterIP | ||||||
|  |   ports: | ||||||
|  |     - port: 80 | ||||||
|  |       targetPort: http | ||||||
|  |       protocol: TCP | ||||||
|  |       name: http | ||||||
|  |   selector: | ||||||
|  |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|  |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     k8s.passbook.io/component: static | ||||||
							
								
								
									
										92
									
								
								helm/passbook/templates/web-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								helm/passbook/templates/web-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,92 @@ | |||||||
|  | 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: | ||||||
|  |       initContainers: | ||||||
|  |         - name: passbook-database-migrations | ||||||
|  |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|  |           command: | ||||||
|  |             - ./manage.py | ||||||
|  |           args: | ||||||
|  |             - migrate | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - 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 | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - 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: 50m | ||||||
|  |               memory: 150M | ||||||
|  |             limits: | ||||||
|  |               cpu: 200m | ||||||
|  |               memory: 300M | ||||||
| @ -1,7 +1,7 @@ | |||||||
| apiVersion: v1 | apiVersion: v1 | ||||||
| kind: Service | kind: Service | ||||||
| metadata: | metadata: | ||||||
|   name: {{ include "passbook.fullname" . }} |   name: {{ include "passbook.fullname" . }}-web | ||||||
|   labels: |   labels: | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|     helm.sh/chart: {{ include "passbook.chart" . }} |     helm.sh/chart: {{ include "passbook.chart" . }} | ||||||
| @ -17,3 +17,4 @@ spec: | |||||||
|   selector: |   selector: | ||||||
|     app.kubernetes.io/name: {{ include "passbook.name" . }} |     app.kubernetes.io/name: {{ include "passbook.name" . }} | ||||||
|     app.kubernetes.io/instance: {{ .Release.Name }} |     app.kubernetes.io/instance: {{ .Release.Name }} | ||||||
|  |     passbook.io/component: web | ||||||
							
								
								
									
										56
									
								
								helm/passbook/templates/worker-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								helm/passbook/templates/worker-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | |||||||
|  | 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: | ||||||
|  |       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 | ||||||
|  |           envFrom: | ||||||
|  |             - configMapRef: | ||||||
|  |                 name: {{ include "passbook.fullname" . }}-config | ||||||
|  |               prefix: PASSBOOK_ | ||||||
|  |           env: | ||||||
|  |             - 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 | ||||||
| @ -5,7 +5,7 @@ | |||||||
| replicaCount: 1 | replicaCount: 1 | ||||||
|  |  | ||||||
| image: | image: | ||||||
|   tag: 0.0.13-alpha |   tag: 0.6.0-beta | ||||||
|  |  | ||||||
| nameOverride: "" | nameOverride: "" | ||||||
|  |  | ||||||
| @ -14,11 +14,20 @@ config: | |||||||
|   # secret_key: _k*@6h2u2@q-dku57hhgzb7tnx*ba9wodcb^s9g0j59@=y(@_o |   # secret_key: _k*@6h2u2@q-dku57hhgzb7tnx*ba9wodcb^s9g0j59@=y(@_o | ||||||
|   # Enable error reporting |   # Enable error reporting | ||||||
|   error_reporting: true |   error_reporting: true | ||||||
|  |   email: | ||||||
|  |     host: localhost | ||||||
|  |  | ||||||
| postgresql: | postgresql: | ||||||
|   postgresqlDatabase: passbook |   postgresqlDatabase: passbook | ||||||
|   postgresqlPassword: foo |   postgresqlPassword: foo | ||||||
|  |  | ||||||
|  | redis: | ||||||
|  |   cluster: | ||||||
|  |     enabled: false | ||||||
|  |   master: | ||||||
|  |     persistence: | ||||||
|  |       enabled: false | ||||||
|  |  | ||||||
| service: | service: | ||||||
|   type: ClusterIP |   type: ClusterIP | ||||||
|   port: 80 |   port: 80 | ||||||
| @ -31,27 +40,7 @@ ingress: | |||||||
|   path: / |   path: / | ||||||
|   hosts: |   hosts: | ||||||
|     - passbook.k8s.local |     - passbook.k8s.local | ||||||
|     - kubernetes-healthcheck-host |  | ||||||
|   defaultHost: passbook.k8s.local |  | ||||||
|   tls: [] |   tls: [] | ||||||
|   #  - secretName: chart-example-tls |   #  - secretName: chart-example-tls | ||||||
|   #    hosts: |   #    hosts: | ||||||
|   #      - passbook.k8s.local |   #      - passbook.k8s.local | ||||||
|  |  | ||||||
| resources: {} |  | ||||||
|   # 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 |  | ||||||
|  |  | ||||||
| nodeSelector: {} |  | ||||||
|  |  | ||||||
| tolerations: [] |  | ||||||
|  |  | ||||||
| affinity: {} |  | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ import os | |||||||
| import sys | import sys | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.core.settings') |     os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'passbook.root.settings') | ||||||
|     try: |     try: | ||||||
|         from django.core.management import execute_from_command_line |         from django.core.management import execute_from_command_line | ||||||
|     except ImportError as exc: |     except ImportError as exc: | ||||||
|  | |||||||
| @ -1,2 +1,2 @@ | |||||||
| """passbook""" | """passbook""" | ||||||
| __version__ = '0.0.13-alpha' | __version__ = '0.6.0-beta' | ||||||
|  | |||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook admin""" |  | ||||||
| __version__ = '0.0.13-alpha' |  | ||||||
|  | |||||||
							
								
								
									
										17
									
								
								passbook/admin/forms/users.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								passbook/admin/forms/users.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | """passbook administrative user forms""" | ||||||
|  |  | ||||||
|  | from django import forms | ||||||
|  |  | ||||||
|  | from passbook.core.models import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UserForm(forms.ModelForm): | ||||||
|  |     """Update User Details""" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         model = User | ||||||
|  |         fields = ['username', 'name', 'email', 'is_staff', 'is_active'] | ||||||
|  |         widgets = { | ||||||
|  |             'name': forms.TextInput | ||||||
|  |         } | ||||||
							
								
								
									
										25
									
								
								passbook/admin/middleware.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								passbook/admin/middleware.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | """passbook admin Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  | from passbook.core.models import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def impersonate(get_response): | ||||||
|  |     """Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  |     def middleware(request): | ||||||
|  |         """Middleware to impersonate users""" | ||||||
|  |  | ||||||
|  |         # User is superuser and has __impersonate ID set | ||||||
|  |         if request.user.is_superuser and "__impersonate" in request.GET: | ||||||
|  |             request.session['impersonate_id'] = request.GET["__impersonate"] | ||||||
|  |         # user wants to stop impersonation | ||||||
|  |         elif "__unimpersonate" in request.GET and 'impersonate_id' in request.session: | ||||||
|  |             del request.session['impersonate_id'] | ||||||
|  |  | ||||||
|  |         # Actually impersonate user | ||||||
|  |         if request.user.is_superuser and 'impersonate_id' in request.session: | ||||||
|  |             request.user = User.objects.get(pk=request.session['impersonate_id']) | ||||||
|  |  | ||||||
|  |         response = get_response(request) | ||||||
|  |         return response | ||||||
|  |     return middleware | ||||||
| @ -1,2 +0,0 @@ | |||||||
| django-rest-framework |  | ||||||
| drf_yasg |  | ||||||
							
								
								
									
										5
									
								
								passbook/admin/settings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								passbook/admin/settings.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | """passbook admin settings""" | ||||||
|  |  | ||||||
|  | MIDDLEWARE = [ | ||||||
|  |     'passbook.admin.middleware.impersonate', | ||||||
|  | ] | ||||||
| @ -4,36 +4,4 @@ | |||||||
| {% load is_active %} | {% load is_active %} | ||||||
|  |  | ||||||
| {% block nav_secondary %} | {% block nav_secondary %} | ||||||
| <ul class="nav navbar-nav navbar-persistent"> |  | ||||||
|   <li class="{% is_active 'passbook_admin:overview' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:overview' %}">{% trans 'Overview' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:applications' 'passbook_admin:application-create' 'passbook_admin:application-update' 'passbook_admin:application-delete' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:applications' %}">{% trans 'Applications' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:sources' 'passbook_admin:source-create' 'passbook_admin:source-update' 'passbook_admin:source-delete' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:sources' %}">{% trans 'Sources' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:providers' 'passbook_admin:provider-create' 'passbook_admin:provider-update' 'passbook_admin:provider-delete' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:providers' %}">{% trans 'Providers' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:factors' 'passbook_admin:factor-create' 'passbook_admin:factor-update' 'passbook_admin:factor-delete' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:factors' %}">{% trans 'Factors' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:policies' 'passbook_admin:policy-create' 'passbook_admin:policy-update' 'passbook_admin:policy-delete' 'passbook_admin:policy-test' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:policies' %}">{% trans 'Policies' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:invitations' 'passbook_admin:invitation-create' 'passbook_admin:invitation-update' 'passbook_admin:invitation-delete' 'passbook_admin:invitation-test' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:invitations' %}">{% trans 'Invitations' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:users' 'passbook_admin:user-update' 'passbook_admin:user-delete' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:users' %}">{% trans 'Users' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active 'passbook_admin:audit-log' %}"> |  | ||||||
|     <a href="{% url 'passbook_admin:audit-log' %}">{% trans 'Audit Log' %}</a> |  | ||||||
|   </li> |  | ||||||
|   <li class="{% is_active_app 'admin' %}"> |  | ||||||
|     <a href="{% url 'admin:index' %}">{% trans 'Django' %}</a> |  | ||||||
|   </li> |  | ||||||
| </ul> |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  | |||||||
							
								
								
									
										31
									
								
								passbook/admin/templates/administration/debug/request.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								passbook/admin/templates/administration/debug/request.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | {% extends "administration/base.html" %} | ||||||
|  |  | ||||||
|  | {% load i18n %} | ||||||
|  | {% load utils %} | ||||||
|  |  | ||||||
|  | {% block title %} | ||||||
|  | {% title %} | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="container"> | ||||||
|  |     <h1><span class="pficon-applications"></span> {% trans "Request" %}</h1> | ||||||
|  |     <hr> | ||||||
|  |     <table class="table table-striped table-bordered"> | ||||||
|  |         <thead> | ||||||
|  |             <tr> | ||||||
|  |                 <th>{% trans 'Key' %}</th> | ||||||
|  |                 <th>{% trans 'Value' %}</th> | ||||||
|  |             </tr> | ||||||
|  |         </thead> | ||||||
|  |         <tbody> | ||||||
|  |             {% for key, value in request_dict.items %} | ||||||
|  |             <tr> | ||||||
|  |                 <td>{{ key }}</td> | ||||||
|  |                 <td>{{ value }}</td> | ||||||
|  |             </tr> | ||||||
|  |             {% endfor %} | ||||||
|  |         </tbody> | ||||||
|  |     </table> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										45
									
								
								passbook/admin/templates/administration/group/list.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								passbook/admin/templates/administration/group/list.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,45 @@ | |||||||
|  | {% extends "administration/base.html" %} | ||||||
|  |  | ||||||
|  | {% load i18n %} | ||||||
|  | {% load utils %} | ||||||
|  |  | ||||||
|  | {% block title %} | ||||||
|  | {% title %} | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="container"> | ||||||
|  |     <h1><span class="pficon-users"></span> {% trans "Groups" %}</h1> | ||||||
|  |     <span>{% trans "Group users together and give them permissions based on the membership." %}</span> | ||||||
|  |     <hr> | ||||||
|  |     <a href="{% url 'passbook_admin:group-create' %}?back={{ request.get_full_path }}" class="btn btn-primary"> | ||||||
|  |         {% trans 'Create...' %} | ||||||
|  |     </a> | ||||||
|  |     <hr> | ||||||
|  |     <table class="table table-striped table-bordered"> | ||||||
|  |         <thead> | ||||||
|  |             <tr> | ||||||
|  |                 <th>{% trans 'Name' %}</th> | ||||||
|  |                 <th>{% trans 'Parent' %}</th> | ||||||
|  |                 <th>{% trans 'Members' %}</th> | ||||||
|  |                 <th></th> | ||||||
|  |             </tr> | ||||||
|  |         </thead> | ||||||
|  |         <tbody> | ||||||
|  |             {% for group in object_list %} | ||||||
|  |             <tr> | ||||||
|  |                 <td>{{ group.name }}</td> | ||||||
|  |                 <td>{{ group.parent }}</td> | ||||||
|  |                 <td>{{ group.user_set.all|length }}</td> | ||||||
|  |                 <td> | ||||||
|  |                     <a class="btn btn-default btn-sm" | ||||||
|  |                         href="{% url 'passbook_admin:group-update' pk=group.uuid %}?back={{ request.get_full_path }}">{% trans 'Edit' %}</a> | ||||||
|  |                     <a class="btn btn-default btn-sm" | ||||||
|  |                         href="{% url 'passbook_admin:group-delete' pk=group.uuid %}?back={{ request.get_full_path }}">{% trans 'Delete' %}</a> | ||||||
|  |                 </td> | ||||||
|  |             </tr> | ||||||
|  |             {% endfor %} | ||||||
|  |         </tbody> | ||||||
|  |     </table> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
| @ -1,83 +0,0 @@ | |||||||
| {% extends "administration/base.html" %} |  | ||||||
|  |  | ||||||
| {% load i18n %} |  | ||||||
| {% load static %} |  | ||||||
| {% load utils %} |  | ||||||
|  |  | ||||||
| {% block head %} |  | ||||||
| {{ block.super }} |  | ||||||
| <link rel="stylesheet" href="{% static 'css/bootstrap-treeview.min.css'%}"> |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block scripts %} |  | ||||||
| {{ block.super }} |  | ||||||
| <script src="{% static 'js/bootstrap-treeview.min.js' %}"></script> |  | ||||||
| <script> |  | ||||||
|   var cleanupData = function (obj) { |  | ||||||
|     return { |  | ||||||
|       text: obj.name, |  | ||||||
|       href: '?group=' + obj.uuid, |  | ||||||
|       nodes: obj.children.map(cleanupData), |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
|  $(function() { |  | ||||||
|    var apiUrl = "{% url 'passbook_admin:group-list' %}?format=json"; |  | ||||||
|    $.ajax({ |  | ||||||
|       url: apiUrl, |  | ||||||
|     }).done(function(data) { |  | ||||||
|       $('#treeview1').treeview({ |  | ||||||
|         collapseIcon: "fa fa-angle-down", |  | ||||||
|         data: data.map(cleanupData), |  | ||||||
|         expandIcon: "fa fa-angle-right", |  | ||||||
|         nodeIcon: "fa pficon-users", |  | ||||||
|         showBorder: true, |  | ||||||
|         enableLinks: true, |  | ||||||
|         onNodeSelected: function (event, node) { |  | ||||||
|           window.location.href = node.href; |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| </script> |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block title %} |  | ||||||
| {% title %} |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="col-md-3"> |  | ||||||
|   <div id="treeview1" class="treeview"> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <div class="col-md-9"> |  | ||||||
|   <h1>{% trans "Invitations" %}</h1> |  | ||||||
|   <a href="{% url 'passbook_admin:invitation-create' %}" class="btn btn-primary"> |  | ||||||
|     {% trans 'Create...' %} |  | ||||||
|   </a> |  | ||||||
|   <hr> |  | ||||||
|   <table class="table table-striped table-bordered"> |  | ||||||
|     <thead> |  | ||||||
|       <tr> |  | ||||||
|         <th>{% trans 'Expiry' %}</th> |  | ||||||
|         <th>{% trans 'Link' %}</th> |  | ||||||
|         <th></th> |  | ||||||
|       </tr> |  | ||||||
|     </thead> |  | ||||||
|     <tbody> |  | ||||||
|       {% for invitation in object_list %} |  | ||||||
|       <tr> |  | ||||||
|         <td>{{ invitation.expires|default:"Never" }}</td> |  | ||||||
|         <td> |  | ||||||
|           <pre>{{ invitation.link }}</pre> |  | ||||||
|         </td> |  | ||||||
|         <td> |  | ||||||
|           <a class="btn btn-default btn-sm" href="{% url 'passbook_admin:invitation-delete' pk=invitation.uuid %}?back={{ request.get_full_path }}">{% |  | ||||||
|             trans 'Delete' %}</a> |  | ||||||
|         </td> |  | ||||||
|       </tr> |  | ||||||
|       {% endfor %} |  | ||||||
|     </tbody> |  | ||||||
|   </table> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @ -76,9 +76,13 @@ | |||||||
|             <div class="card-pf-body"> |             <div class="card-pf-body"> | ||||||
|                 <p class="card-pf-aggregate-status-notifications"> |                 <p class="card-pf-aggregate-status-notifications"> | ||||||
|                     <span class="card-pf-aggregate-status-notification"> |                     <span class="card-pf-aggregate-status-notification"> | ||||||
|                         <a href="{% url 'passbook_admin:factors' %}"> |                         {% if factor_count < 1 %} | ||||||
|  |                         <span class="pficon-error-circle-o" data-toggle="tooltip" data-placement="right" | ||||||
|  |                             title="{% trans 'No Factors configured. No Users will be able to login.' %}"></span> | ||||||
|  |                         {{ factor_count }} | ||||||
|  |                         {% else %} | ||||||
|                         <span class="pficon pficon-ok"></span>{{ factor_count }} |                         <span class="pficon pficon-ok"></span>{{ factor_count }} | ||||||
|                         </a> |                         {% endif %} | ||||||
|                     </span> |                     </span> | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
| @ -95,9 +99,13 @@ | |||||||
|             <div class="card-pf-body"> |             <div class="card-pf-body"> | ||||||
|                 <p class="card-pf-aggregate-status-notifications"> |                 <p class="card-pf-aggregate-status-notifications"> | ||||||
|                     <span class="card-pf-aggregate-status-notification"> |                     <span class="card-pf-aggregate-status-notification"> | ||||||
|                         <a href="{% url 'passbook_admin:policies' %}"> |                         {% if policies_without_attachment > 0 %} | ||||||
|  |                         <span class="pficon-warning-triangle-o" data-toggle="tooltip" data-placement="right" | ||||||
|  |                             title="{% trans 'Policies without attachment exist.' %}"></span> | ||||||
|  |                         {{ policy_count }} | ||||||
|  |                         {% else %} | ||||||
|                         <span class="pficon pficon-ok"></span>{{ policy_count }} |                         <span class="pficon pficon-ok"></span>{{ policy_count }} | ||||||
|                         </a> |                         {% endif %} | ||||||
|                     </span> |                     </span> | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
| @ -144,10 +152,8 @@ | |||||||
|     <div class="col-xs-6 col-sm-2 col-md-2"> |     <div class="col-xs-6 col-sm-2 col-md-2"> | ||||||
|         <div class="card-pf card-pf-accented card-pf-aggregate-status"> |         <div class="card-pf card-pf-accented card-pf-aggregate-status"> | ||||||
|             <h2 class="card-pf-title"> |             <h2 class="card-pf-title"> | ||||||
|                 <a href="#"> |  | ||||||
|                 <span class="pficon-bundle"></span> |                 <span class="pficon-bundle"></span> | ||||||
|                 <span class="card-pf-aggregate-status-count"></span> {% trans 'Version' %} |                 <span class="card-pf-aggregate-status-count"></span> {% trans 'Version' %} | ||||||
|                 </a> |  | ||||||
|             </h2> |             </h2> | ||||||
|             <div class="card-pf-body"> |             <div class="card-pf-body"> | ||||||
|                 <p class="card-pf-aggregate-status-notifications"> |                 <p class="card-pf-aggregate-status-notifications"> | ||||||
| @ -173,8 +179,8 @@ | |||||||
|                     <span class="card-pf-aggregate-status-notification"> |                     <span class="card-pf-aggregate-status-notification"> | ||||||
|                         <a href="#"> |                         <a href="#"> | ||||||
|                             {% if worker_count < 1%} |                             {% if worker_count < 1%} | ||||||
|                             <span class="pficon-error-circle-o" data-toggle="tooltip" data-placement="right" |                             <span class="pficon-warning-triangle-o" data-toggle="tooltip" data-placement="right" | ||||||
|                                 title="{% trans 'No workers connected. Policies may not work.' %}"></span> {{ worker_count }} |                                 title="{% trans 'No workers connected.' %}"></span> {{ worker_count }} | ||||||
|                             {% else %} |                             {% else %} | ||||||
|                             <span class="pficon pficon-ok"></span>{{ worker_count }} |                             <span class="pficon pficon-ok"></span>{{ worker_count }} | ||||||
|                             {% endif %} |                             {% endif %} | ||||||
| @ -184,5 +190,59 @@ | |||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  |     <div class="col-xs-6 col-sm-2 col-md-2"> | ||||||
|  |         <div class="card-pf card-pf-accented card-pf-aggregate-status"> | ||||||
|  |             <h2 class="card-pf-title"> | ||||||
|  |                 <span class="pficon-server"></span> | ||||||
|  |                 <span class="card-pf-aggregate-status-count"></span> {% trans 'Cached Policies' %} | ||||||
|  |             </h2> | ||||||
|  |             <div class="card-pf-body"> | ||||||
|  |                 <p class="card-pf-aggregate-status-notifications"> | ||||||
|  |                     <span class="card-pf-aggregate-status-notification"> | ||||||
|  |                         <a href="#" data-toggle="modal" data-target="#clearCacheMOdal"> | ||||||
|  |                             {% if cached_policies < 1 %} | ||||||
|  |                             <span class="pficon-warning-triangle-o" data-toggle="tooltip" data-placement="right" | ||||||
|  |                                 title="{% trans 'No policies cached. Users may experience slow response times.' %}"></span> {{ cached_policies }} | ||||||
|  |                             {% else %} | ||||||
|  |                             <span class="pficon pficon-ok"></span>{{ cached_policies }} | ||||||
|  |                             {% endif %} | ||||||
|  |                         </a> | ||||||
|  |                     </span> | ||||||
|  |                 </p> | ||||||
|  |             </div> | ||||||
|  |         </div> | ||||||
|  |     </div> | ||||||
|  | </div> | ||||||
|  | <div class="modal fade" id="clearCacheMOdal" tabindex="-1" role="dialog" aria-labelledby="clearCacheMOdalLabel" aria-hidden="true"> | ||||||
|  |   <div class="modal-dialog"> | ||||||
|  |     <div class="modal-content"> | ||||||
|  |       <div class="modal-header"> | ||||||
|  |         <button type="button" class="close" data-dismiss="modal" aria-hidden="true"> | ||||||
|  |           <span class="pficon pficon-close"></span> | ||||||
|  |         </button> | ||||||
|  |         <h4 class="modal-title" id="clearCacheMOdalLabel">{% trans 'Clear Cache' %}</h4> | ||||||
|  |       </div> | ||||||
|  |       <div class="modal-body"> | ||||||
|  |         <form method="post" id="clearForm"> | ||||||
|  |             {% csrf_token %} | ||||||
|  |             <input type="hidden" name="clear"> | ||||||
|  |             <p> | ||||||
|  |                 {% blocktrans %} | ||||||
|  |                     Are you sure you want to clear the cache? This includes all user sessions and all cached Policy results. | ||||||
|  |                 {% endblocktrans %} | ||||||
|  |             </p> | ||||||
|  |             <h3> | ||||||
|  |                 {% blocktrans %} | ||||||
|  |                     This will also log you out. | ||||||
|  |                 {% endblocktrans %} | ||||||
|  |             </h3> | ||||||
|  |         </form> | ||||||
|  |       </div> | ||||||
|  |       <div class="modal-footer"> | ||||||
|  |         <button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button> | ||||||
|  |         <button form="clearForm" type="submit" type="button" class="btn btn-danger">{% trans 'Clear' %}</button> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
| </div> | </div> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  | |||||||
| @ -28,6 +28,7 @@ | |||||||
|     <table class="table table-striped table-bordered"> |     <table class="table table-striped table-bordered"> | ||||||
|         <thead> |         <thead> | ||||||
|             <tr> |             <tr> | ||||||
|  |                 <th></th> | ||||||
|                 <th>{% trans 'Name' %}</th> |                 <th>{% trans 'Name' %}</th> | ||||||
|                 <th>{% trans 'Type' %}</th> |                 <th>{% trans 'Type' %}</th> | ||||||
|                 <th></th> |                 <th></th> | ||||||
| @ -35,7 +36,14 @@ | |||||||
|         </thead> |         </thead> | ||||||
|         <tbody> |         <tbody> | ||||||
|             {% for policy in object_list %} |             {% for policy in object_list %} | ||||||
|             <tr> |             <tr {% if not policy.policymodel_set.exists %} class="warning" {% endif %}> | ||||||
|  |                 <th> | ||||||
|  |                     {% if not policy.policymodel_set.exists %} | ||||||
|  |                     <span class="pficon-warning-triangle-o" data-toggle="tooltip" data-placement="right" title="{% trans 'Warning: Policy is not assigned.' %}"></span> | ||||||
|  |                     {% else %} | ||||||
|  |                     <span class="pficon-ok" data-toggle="tooltip" data-placement="right" title="{% blocktrans with objects=policy.policymodel_set.all|join:', ' %}Assigned to objects {{ objects }}{% endblocktrans %}"></span> | ||||||
|  |                     {% endif %} | ||||||
|  |                 </th> | ||||||
|                 <td>{{ policy.name }}</td> |                 <td>{{ policy.name }}</td> | ||||||
|                 <td>{{ policy|verbose_name }}</td> |                 <td>{{ policy|verbose_name }}</td> | ||||||
|                 <td> |                 <td> | ||||||
|  | |||||||
| @ -5,3 +5,22 @@ | |||||||
| {% block above_form %} | {% block above_form %} | ||||||
| <h1>{% blocktrans with policy=policy %}Test policy {{ policy }}{% endblocktrans %}</h1> | <h1>{% blocktrans with policy=policy %}Test policy {{ policy }}{% endblocktrans %}</h1> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block action %} | ||||||
|  | {% trans 'Test' %} | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block beneath_form %} | ||||||
|  | <p class="loading" style="display: none;"> | ||||||
|  |   <span class="spinner spinner-xs spinner-inline"></span> {% trans 'Processing, please wait...' %} | ||||||
|  | </p> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block scripts %} | ||||||
|  | {{ block.super }} | ||||||
|  | <script> | ||||||
|  |   $('form').on('submit', function () { | ||||||
|  |     $('p.loading').show(); | ||||||
|  |   }) | ||||||
|  | </script> | ||||||
|  | {% endblock %} | ||||||
|  | |||||||
| @ -0,0 +1,52 @@ | |||||||
|  | {% extends "administration/base.html" %} | ||||||
|  |  | ||||||
|  | {% load i18n %} | ||||||
|  | {% load utils %} | ||||||
|  |  | ||||||
|  | {% block title %} | ||||||
|  | {% title %} | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="container"> | ||||||
|  |     <h1><span class="fa fa-table"></span> {% trans "Property Mappings" %}</h1> | ||||||
|  |     <span>{% trans "Property Mappings allow you expose provider-specific attributes." %}</span> | ||||||
|  |     <hr> | ||||||
|  |     <div class="dropdown"> | ||||||
|  |         <button class="btn btn-primary dropdown-toggle" type="button" id="createDropdown" data-toggle="dropdown"> | ||||||
|  |             {% trans 'Create...' %} | ||||||
|  |             <span class="caret"></span> | ||||||
|  |         </button> | ||||||
|  |         <ul class="dropdown-menu" role="menu" aria-labelledby="createDropdown"> | ||||||
|  |             {% for type, name in types.items %} | ||||||
|  |             <li role="presentation"><a role="menuitem" tabindex="-1" | ||||||
|  |                     href="{% url 'passbook_admin:property-mapping-create' %}?type={{ type }}&back={{ request.get_full_path }}">{{ name }}</a></li> | ||||||
|  |             {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |     </div> | ||||||
|  |     <hr> | ||||||
|  |     <table class="table table-striped table-bordered"> | ||||||
|  |         <thead> | ||||||
|  |             <tr> | ||||||
|  |                 <th>{% trans 'Name' %}</th> | ||||||
|  |                 <th>{% trans 'Type' %}</th> | ||||||
|  |                 <th></th> | ||||||
|  |             </tr> | ||||||
|  |         </thead> | ||||||
|  |         <tbody> | ||||||
|  |             {% for property_mapping in object_list %} | ||||||
|  |             <tr> | ||||||
|  |                 <td>{{ property_mapping.name }}</td> | ||||||
|  |                 <td>{{ property_mapping|verbose_name }}</td> | ||||||
|  |                 <td> | ||||||
|  |                     <a class="btn btn-default btn-sm" | ||||||
|  |                         href="{% url 'passbook_admin:property-mapping-update' pk=property_mapping.pk %}?back={{ request.get_full_path }}">{% trans 'Edit' %}</a> | ||||||
|  |                     <a class="btn btn-default btn-sm" | ||||||
|  |                         href="{% url 'passbook_admin:property-mapping-delete' pk=property_mapping.pk %}?back={{ request.get_full_path }}">{% trans 'Delete' %}</a> | ||||||
|  |                 </td> | ||||||
|  |             </tr> | ||||||
|  |             {% endfor %} | ||||||
|  |         </tbody> | ||||||
|  |     </table> | ||||||
|  | </div> | ||||||
|  | {% endblock %} | ||||||
| @ -57,6 +57,10 @@ | |||||||
|                     <a class="btn btn-default btn-sm" |                     <a class="btn btn-default btn-sm" | ||||||
|                         href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a> |                         href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a> | ||||||
|                     {% endfor %} |                     {% endfor %} | ||||||
|  |                     {% get_htmls provider as htmls %} | ||||||
|  |                     {% for html in htmls %} | ||||||
|  |                     {{ html|safe }} | ||||||
|  |                     {% endfor %} | ||||||
|                 </td> |                 </td> | ||||||
|             </tr> |             </tr> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|  | |||||||
| @ -36,7 +36,7 @@ | |||||||
|             <tr> |             <tr> | ||||||
|                 <td>{{ source.name }}</td> |                 <td>{{ source.name }}</td> | ||||||
|                 <td>{{ source|fieldtype }}</td> |                 <td>{{ source|fieldtype }}</td> | ||||||
|                 <td>{{ source.additional_info }}</td> |                 <td>{{ source.additional_info|safe }}</td> | ||||||
|                 <td> |                 <td> | ||||||
|                     <a class="btn btn-default btn-sm" |                     <a class="btn btn-default btn-sm" | ||||||
|                         href="{% url 'passbook_admin:source-update' pk=source.uuid %}?back={{ request.get_full_path }}">{% trans 'Edit' %}</a> |                         href="{% url 'passbook_admin:source-update' pk=source.uuid %}?back={{ request.get_full_path }}">{% trans 'Edit' %}</a> | ||||||
|  | |||||||
| @ -31,6 +31,8 @@ | |||||||
|                         href="{% url 'passbook_admin:user-delete' pk=user.pk %}?back={{ request.get_full_path }}">{% trans 'Delete' %}</a> |                         href="{% url 'passbook_admin:user-delete' pk=user.pk %}?back={{ request.get_full_path }}">{% trans 'Delete' %}</a> | ||||||
|                     <a class="btn btn-default btn-sm" |                     <a class="btn btn-default btn-sm" | ||||||
|                         href="{% url 'passbook_admin:user-password-reset' pk=user.pk %}?back={{ request.get_full_path }}">{% trans 'Reset Password' %}</a> |                         href="{% url 'passbook_admin:user-password-reset' pk=user.pk %}?back={{ request.get_full_path }}">{% trans 'Reset Password' %}</a> | ||||||
|  |                     <a class="btn btn-default btn-sm" | ||||||
|  |                         href="{% url 'passbook_core:overview' %}?__impersonate={{ user.pk }}">{% trans 'Impersonate' %}</a> | ||||||
|                 </td> |                 </td> | ||||||
|             </tr> |             </tr> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|  | |||||||
| @ -2,6 +2,21 @@ | |||||||
|  |  | ||||||
| {% load i18n %} | {% load i18n %} | ||||||
| {% load utils %} | {% load utils %} | ||||||
|  | {% load static %} | ||||||
|  |  | ||||||
|  | {% block head %} | ||||||
|  | {{ block.super }} | ||||||
|  | {{ form.media.css }} | ||||||
|  | <script type="text/javascript" src="{% url 'admin:jsi18n' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/vendor/jquery/jquery.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/jquery.init.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/core.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/actions.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/urlify.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/prepopulate.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/SelectBox.js' %}"></script> | ||||||
|  | <script type="text/javascript" src="{% static 'admin/js/SelectFilter2.js' %}"></script> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| <div class="container"> | <div class="container"> | ||||||
| @ -14,5 +29,12 @@ | |||||||
|       <input type="submit" class="btn btn-primary" value="{% block action %}{% endblock %}" /> |       <input type="submit" class="btn btn-primary" value="{% block action %}{% endblock %}" /> | ||||||
|     </form> |     </form> | ||||||
|   </div> |   </div> | ||||||
|  |   {% block beneath_form %} | ||||||
|  |   {% endblock %} | ||||||
| </div> | </div> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block scripts %} | ||||||
|  | {{ block.super }} | ||||||
|  | {{ form.media.js }} | ||||||
|  | {% endblock %} | ||||||
|  | |||||||
| @ -1,12 +1,14 @@ | |||||||
| """passbook admin templatetags""" | """passbook admin templatetags""" | ||||||
| import inspect | import inspect | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| from django import template | from django import template | ||||||
| from django.db.models import Model | from django.db.models import Model | ||||||
|  | from structlog import get_logger | ||||||
|  |  | ||||||
|  | from passbook.lib.utils.template import render_to_string | ||||||
|  |  | ||||||
| register = template.Library() | register = template.Library() | ||||||
| LOGGER = getLogger(__name__) | LOGGER = get_logger() | ||||||
|  |  | ||||||
| @register.simple_tag() | @register.simple_tag() | ||||||
| def get_links(model_instance): | def get_links(model_instance): | ||||||
| @ -29,3 +31,24 @@ def get_links(model_instance): | |||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     return links |     return links | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @register.simple_tag(takes_context=True) | ||||||
|  | def get_htmls(context, model_instance): | ||||||
|  |     """Find all html_ methods on an object instance, run them and return as dict""" | ||||||
|  |     prefix = 'html_' | ||||||
|  |     htmls = [] | ||||||
|  |  | ||||||
|  |     if not isinstance(model_instance, Model): | ||||||
|  |         LOGGER.warning("Model %s is not instance of Model", model_instance) | ||||||
|  |         return htmls | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         for name, method in inspect.getmembers(model_instance, predicate=inspect.ismethod): | ||||||
|  |             if name.startswith(prefix): | ||||||
|  |                 template, _context = method(context.get('request')) | ||||||
|  |                 htmls.append(render_to_string(template, _context)) | ||||||
|  |     except NotImplementedError: | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     return htmls | ||||||
|  | |||||||
| @ -1,9 +1,9 @@ | |||||||
| """passbook URL Configuration""" | """passbook URL Configuration""" | ||||||
| from django.urls import include, path | from django.urls import include, path | ||||||
|  |  | ||||||
| from passbook.admin.views import (applications, audit, factors, groups, | from passbook.admin.views import (applications, audit, debug, factors, groups, | ||||||
|                                   invitations, overview, policy, providers, |                                   invitations, overview, policy, | ||||||
|                                   sources, users) |                                   property_mapping, providers, sources, users) | ||||||
|  |  | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     path('', overview.AdministrationOverviewView.as_view(), name='overview'), |     path('', overview.AdministrationOverviewView.as_view(), name='overview'), | ||||||
| @ -43,6 +43,15 @@ urlpatterns = [ | |||||||
|          factors.FactorUpdateView.as_view(), name='factor-update'), |          factors.FactorUpdateView.as_view(), name='factor-update'), | ||||||
|     path('factors/<uuid:pk>/delete/', |     path('factors/<uuid:pk>/delete/', | ||||||
|          factors.FactorDeleteView.as_view(), name='factor-delete'), |          factors.FactorDeleteView.as_view(), name='factor-delete'), | ||||||
|  |     # Factors | ||||||
|  |     path('property-mappings/', property_mapping.PropertyMappingListView.as_view(), | ||||||
|  |          name='property-mappings'), | ||||||
|  |     path('property-mappings/create/', | ||||||
|  |          property_mapping.PropertyMappingCreateView.as_view(), name='property-mapping-create'), | ||||||
|  |     path('property-mappings/<uuid:pk>/update/', | ||||||
|  |          property_mapping.PropertyMappingUpdateView.as_view(), name='property-mapping-update'), | ||||||
|  |     path('property-mappings/<uuid:pk>/delete/', | ||||||
|  |          property_mapping.PropertyMappingDeleteView.as_view(), name='property-mapping-delete'), | ||||||
|     # Invitations |     # Invitations | ||||||
|     path('invitations/', invitations.InvitationListView.as_view(), name='invitations'), |     path('invitations/', invitations.InvitationListView.as_view(), name='invitations'), | ||||||
|     path('invitations/create/', |     path('invitations/create/', | ||||||
| @ -58,10 +67,17 @@ urlpatterns = [ | |||||||
|          users.UserDeleteView.as_view(), name='user-delete'), |          users.UserDeleteView.as_view(), name='user-delete'), | ||||||
|     path('users/<int:pk>/reset/', |     path('users/<int:pk>/reset/', | ||||||
|          users.UserPasswordResetView.as_view(), name='user-password-reset'), |          users.UserPasswordResetView.as_view(), name='user-password-reset'), | ||||||
|  |     # Groups | ||||||
|  |     path('group/', groups.GroupListView.as_view(), name='group'), | ||||||
|  |     path('group/create/', groups.GroupCreateView.as_view(), name='group-create'), | ||||||
|  |     path('group/<uuid:pk>/update/', groups.GroupUpdateView.as_view(), name='group-update'), | ||||||
|  |     path('group/<uuid:pk>/delete/', groups.GroupDeleteView.as_view(), name='group-delete'), | ||||||
|     # Audit Log |     # Audit Log | ||||||
|     path('audit/', audit.AuditEntryListView.as_view(), name='audit-log'), |     path('audit/', audit.AuditEntryListView.as_view(), name='audit-log'), | ||||||
|     # Groups |     # Groups | ||||||
|     path('groups/', groups.GroupListView.as_view(), name='groups'), |     path('groups/', groups.GroupListView.as_view(), name='groups'), | ||||||
|     # API |     # API | ||||||
|     path('api/', include('passbook.admin.api.urls')) |     path('api/', include('passbook.admin.api.urls')), | ||||||
|  |     # Debug | ||||||
|  |     path('debug/request/', debug.DebugRequestView.as_view(), name='debug-request'), | ||||||
| ] | ] | ||||||
|  | |||||||
							
								
								
									
										17
									
								
								passbook/admin/views/debug.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								passbook/admin/views/debug.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | """passbook administration debug views""" | ||||||
|  |  | ||||||
|  | from django.views.generic import TemplateView | ||||||
|  |  | ||||||
|  | from passbook.admin.mixins import AdminRequiredMixin | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DebugRequestView(AdminRequiredMixin, TemplateView): | ||||||
|  |     """Show debug info about request""" | ||||||
|  |  | ||||||
|  |     template_name = 'administration/debug/request.html' | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         kwargs['request_dict'] = {} | ||||||
|  |         for key in dir(self.request): | ||||||
|  |             kwargs['request_dict'][key] = getattr(self.request, key) | ||||||
|  |         return super().get_context_data(**kwargs) | ||||||
| @ -1,12 +1,57 @@ | |||||||
| """passbook Group administration""" | """passbook Group administration""" | ||||||
| from django.views.generic import ListView | from django.contrib import messages | ||||||
|  | from django.contrib.messages.views import SuccessMessageMixin | ||||||
|  | from django.urls import reverse_lazy | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  | from django.views.generic import CreateView, DeleteView, ListView, UpdateView | ||||||
|  |  | ||||||
| from passbook.admin.mixins import AdminRequiredMixin | from passbook.admin.mixins import AdminRequiredMixin | ||||||
|  | from passbook.core.forms.groups import GroupForm | ||||||
| from passbook.core.models import Group | from passbook.core.models import Group | ||||||
|  |  | ||||||
|  |  | ||||||
| class GroupListView(AdminRequiredMixin, ListView): | class GroupListView(AdminRequiredMixin, ListView): | ||||||
|     """Show list of all invitations""" |     """Show list of all groups""" | ||||||
|  |  | ||||||
|     model = Group |     model = Group | ||||||
|     template_name = 'administration/groups/list.html' |     ordering = 'name' | ||||||
|  |     template_name = 'administration/group/list.html' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GroupCreateView(SuccessMessageMixin, AdminRequiredMixin, CreateView): | ||||||
|  |     """Create new Group""" | ||||||
|  |  | ||||||
|  |     form_class = GroupForm | ||||||
|  |  | ||||||
|  |     template_name = 'generic/create.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:groups') | ||||||
|  |     success_message = _('Successfully created Group') | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         kwargs['type'] = 'Group' | ||||||
|  |         return super().get_context_data(**kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GroupUpdateView(SuccessMessageMixin, AdminRequiredMixin, UpdateView): | ||||||
|  |     """Update group""" | ||||||
|  |  | ||||||
|  |     model = Group | ||||||
|  |     form_class = GroupForm | ||||||
|  |  | ||||||
|  |     template_name = 'generic/update.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:groups') | ||||||
|  |     success_message = _('Successfully updated Group') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GroupDeleteView(SuccessMessageMixin, AdminRequiredMixin, DeleteView): | ||||||
|  |     """Delete group""" | ||||||
|  |  | ||||||
|  |     model = Group | ||||||
|  |  | ||||||
|  |     template_name = 'generic/delete.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:groups') | ||||||
|  |     success_message = _('Successfully deleted Group') | ||||||
|  |  | ||||||
|  |     def delete(self, request, *args, **kwargs): | ||||||
|  |         messages.success(self.request, self.success_message) | ||||||
|  |         return super().delete(request, *args, **kwargs) | ||||||
|  | |||||||
| @ -1,11 +1,13 @@ | |||||||
| """passbook administration overview""" | """passbook administration overview""" | ||||||
|  | from django.core.cache import cache | ||||||
|  | from django.shortcuts import redirect, reverse | ||||||
| from django.views.generic import TemplateView | from django.views.generic import TemplateView | ||||||
|  |  | ||||||
|  | from passbook import __version__ | ||||||
| from passbook.admin.mixins import AdminRequiredMixin | from passbook.admin.mixins import AdminRequiredMixin | ||||||
| from passbook.core import __version__ |  | ||||||
| from passbook.core.celery import CELERY_APP |  | ||||||
| from passbook.core.models import (Application, Factor, Invitation, Policy, | from passbook.core.models import (Application, Factor, Invitation, Policy, | ||||||
|                                   Provider, Source, User) |                                   Provider, Source, User) | ||||||
|  | from passbook.root.celery import CELERY_APP | ||||||
|  |  | ||||||
|  |  | ||||||
| class AdministrationOverviewView(AdminRequiredMixin, TemplateView): | class AdministrationOverviewView(AdminRequiredMixin, TemplateView): | ||||||
| @ -13,6 +15,13 @@ class AdministrationOverviewView(AdminRequiredMixin, TemplateView): | |||||||
|  |  | ||||||
|     template_name = 'administration/overview.html' |     template_name = 'administration/overview.html' | ||||||
|  |  | ||||||
|  |     def post(self, *args, **kwargs): | ||||||
|  |         """Handle post (clear cache from modal)""" | ||||||
|  |         if 'clear' in self.request.POST: | ||||||
|  |             cache.clear() | ||||||
|  |             return redirect(reverse('passbook_core:auth-login')) | ||||||
|  |         return self.get(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs['application_count'] = len(Application.objects.all()) |         kwargs['application_count'] = len(Application.objects.all()) | ||||||
|         kwargs['policy_count'] = len(Policy.objects.all()) |         kwargs['policy_count'] = len(Policy.objects.all()) | ||||||
| @ -24,4 +33,6 @@ class AdministrationOverviewView(AdminRequiredMixin, TemplateView): | |||||||
|         kwargs['version'] = __version__ |         kwargs['version'] = __version__ | ||||||
|         kwargs['worker_count'] = len(CELERY_APP.control.ping(timeout=0.5)) |         kwargs['worker_count'] = len(CELERY_APP.control.ping(timeout=0.5)) | ||||||
|         kwargs['providers_without_application'] = Provider.objects.filter(application=None) |         kwargs['providers_without_application'] = Provider.objects.filter(application=None) | ||||||
|  |         kwargs['policies_without_attachment'] = len(Policy.objects.filter(policymodel__isnull=True)) | ||||||
|  |         kwargs['cached_policies'] = len(cache.keys('policy_*')) | ||||||
|         return super().get_context_data(**kwargs) |         return super().get_context_data(**kwargs) | ||||||
|  | |||||||
| @ -12,6 +12,7 @@ from passbook.admin.forms.policies import PolicyTestForm | |||||||
| from passbook.admin.mixins import AdminRequiredMixin | from passbook.admin.mixins import AdminRequiredMixin | ||||||
| from passbook.core.models import Policy | from passbook.core.models import Policy | ||||||
| from passbook.lib.utils.reflection import path_to_class | from passbook.lib.utils.reflection import path_to_class | ||||||
|  | from passbook.policies.engine import PolicyEngine | ||||||
|  |  | ||||||
|  |  | ||||||
| class PolicyListView(AdminRequiredMixin, ListView): | class PolicyListView(AdminRequiredMixin, ListView): | ||||||
| @ -100,7 +101,9 @@ class PolicyTestView(AdminRequiredMixin, DetailView, FormView): | |||||||
|     def form_valid(self, form): |     def form_valid(self, form): | ||||||
|         policy = self.get_object() |         policy = self.get_object() | ||||||
|         user = form.cleaned_data.get('user') |         user = form.cleaned_data.get('user') | ||||||
|         result = policy.passes(user) |         policy_engine = PolicyEngine([policy]) | ||||||
|  |         policy_engine.for_user(user).with_request(self.request).build() | ||||||
|  |         result = policy_engine.passing | ||||||
|         if result: |         if result: | ||||||
|             messages.success(self.request, _('User successfully passed policy.')) |             messages.success(self.request, _('User successfully passed policy.')) | ||||||
|         else: |         else: | ||||||
|  | |||||||
							
								
								
									
										90
									
								
								passbook/admin/views/property_mapping.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								passbook/admin/views/property_mapping.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,90 @@ | |||||||
|  | """passbook PropertyMapping administration""" | ||||||
|  | from django.contrib import messages | ||||||
|  | from django.contrib.messages.views import SuccessMessageMixin | ||||||
|  | from django.http import Http404 | ||||||
|  | from django.urls import reverse_lazy | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  | from django.views.generic import CreateView, DeleteView, ListView, UpdateView | ||||||
|  |  | ||||||
|  | from passbook.admin.mixins import AdminRequiredMixin | ||||||
|  | from passbook.core.models import PropertyMapping | ||||||
|  | from passbook.lib.utils.reflection import path_to_class | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def all_subclasses(cls): | ||||||
|  |     """Recursively return all subclassess of cls""" | ||||||
|  |     return set(cls.__subclasses__()).union( | ||||||
|  |         [s for c in cls.__subclasses__() for s in all_subclasses(c)]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PropertyMappingListView(AdminRequiredMixin, ListView): | ||||||
|  |     """Show list of all property_mappings""" | ||||||
|  |  | ||||||
|  |     model = PropertyMapping | ||||||
|  |     template_name = 'administration/property_mapping/list.html' | ||||||
|  |     ordering = 'name' | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         kwargs['types'] = { | ||||||
|  |             x.__name__: x._meta.verbose_name for x in all_subclasses(PropertyMapping)} | ||||||
|  |         return super().get_context_data(**kwargs) | ||||||
|  |  | ||||||
|  |     def get_queryset(self): | ||||||
|  |         return super().get_queryset().select_subclasses() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PropertyMappingCreateView(SuccessMessageMixin, AdminRequiredMixin, CreateView): | ||||||
|  |     """Create new PropertyMapping""" | ||||||
|  |  | ||||||
|  |     template_name = 'generic/create.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:property-mappings') | ||||||
|  |     success_message = _('Successfully created Property Mapping') | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         kwargs = super().get_context_data(**kwargs) | ||||||
|  |         property_mapping_type = self.request.GET.get('type') | ||||||
|  |         model = next(x for x in all_subclasses(PropertyMapping) | ||||||
|  |                      if x.__name__ == property_mapping_type) | ||||||
|  |         kwargs['type'] = model._meta.verbose_name | ||||||
|  |         return kwargs | ||||||
|  |  | ||||||
|  |     def get_form_class(self): | ||||||
|  |         property_mapping_type = self.request.GET.get('type') | ||||||
|  |         model = next(x for x in all_subclasses(PropertyMapping) | ||||||
|  |                      if x.__name__ == property_mapping_type) | ||||||
|  |         if not model: | ||||||
|  |             raise Http404 | ||||||
|  |         return path_to_class(model.form) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PropertyMappingUpdateView(SuccessMessageMixin, AdminRequiredMixin, UpdateView): | ||||||
|  |     """Update property_mapping""" | ||||||
|  |  | ||||||
|  |     model = PropertyMapping | ||||||
|  |     template_name = 'generic/update.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:property-mappings') | ||||||
|  |     success_message = _('Successfully updated Property Mapping') | ||||||
|  |  | ||||||
|  |     def get_form_class(self): | ||||||
|  |         form_class_path = self.get_object().form | ||||||
|  |         form_class = path_to_class(form_class_path) | ||||||
|  |         return form_class | ||||||
|  |  | ||||||
|  |     def get_object(self, queryset=None): | ||||||
|  |         return PropertyMapping.objects.filter(pk=self.kwargs.get('pk')).select_subclasses().first() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PropertyMappingDeleteView(SuccessMessageMixin, AdminRequiredMixin, DeleteView): | ||||||
|  |     """Delete property_mapping""" | ||||||
|  |  | ||||||
|  |     model = PropertyMapping | ||||||
|  |     template_name = 'generic/delete.html' | ||||||
|  |     success_url = reverse_lazy('passbook_admin:property-mappings') | ||||||
|  |     success_message = _('Successfully deleted Property Mapping') | ||||||
|  |  | ||||||
|  |     def get_object(self, queryset=None): | ||||||
|  |         return PropertyMapping.objects.filter(pk=self.kwargs.get('pk')).select_subclasses().first() | ||||||
|  |  | ||||||
|  |     def delete(self, request, *args, **kwargs): | ||||||
|  |         messages.success(self.request, self.success_message) | ||||||
|  |         return super().delete(request, *args, **kwargs) | ||||||
| @ -7,8 +7,8 @@ from django.utils.translation import ugettext as _ | |||||||
| from django.views import View | from django.views import View | ||||||
| from django.views.generic import DeleteView, ListView, UpdateView | from django.views.generic import DeleteView, ListView, UpdateView | ||||||
|  |  | ||||||
|  | from passbook.admin.forms.users import UserForm | ||||||
| from passbook.admin.mixins import AdminRequiredMixin | from passbook.admin.mixins import AdminRequiredMixin | ||||||
| from passbook.core.forms.users import UserDetailForm |  | ||||||
| from passbook.core.models import Nonce, User | from passbook.core.models import Nonce, User | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -23,7 +23,7 @@ class UserUpdateView(SuccessMessageMixin, AdminRequiredMixin, UpdateView): | |||||||
|     """Update user""" |     """Update user""" | ||||||
|  |  | ||||||
|     model = User |     model = User | ||||||
|     form_class = UserDetailForm |     form_class = UserForm | ||||||
|  |  | ||||||
|     template_name = 'generic/update.html' |     template_name = 'generic/update.html' | ||||||
|     success_url = reverse_lazy('passbook_admin:users') |     success_url = reverse_lazy('passbook_admin:users') | ||||||
|  | |||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook api""" |  | ||||||
| __version__ = '0.0.13-alpha' |  | ||||||
|  | |||||||
| @ -1,3 +0,0 @@ | |||||||
| django-rest-framework |  | ||||||
| drf_yasg |  | ||||||
| django-filters |  | ||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook audit Header""" |  | ||||||
| __version__ = '0.0.13-alpha' |  | ||||||
|  | |||||||
| @ -1,7 +1,8 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-16 09:13 | # Generated by Django 2.2.6 on 2019-10-07 14:07 | ||||||
|  |  | ||||||
| import uuid | import uuid | ||||||
|  |  | ||||||
|  | import django.contrib.postgres.fields.jsonb | ||||||
| import django.db.models.deletion | import django.db.models.deletion | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db import migrations, models | from django.db import migrations, models | ||||||
| @ -23,7 +24,7 @@ class Migration(migrations.Migration): | |||||||
|                 ('action', models.TextField(choices=[('login', 'login'), ('login_failed', 'login_failed'), ('logout', 'logout'), ('authorize_application', 'authorize_application'), ('suspicious_request', 'suspicious_request'), ('sign_up', 'sign_up'), ('password_reset', 'password_reset'), ('invitation_created', 'invitation_created'), ('invitation_used', 'invitation_used')])), |                 ('action', models.TextField(choices=[('login', 'login'), ('login_failed', 'login_failed'), ('logout', 'logout'), ('authorize_application', 'authorize_application'), ('suspicious_request', 'suspicious_request'), ('sign_up', 'sign_up'), ('password_reset', 'password_reset'), ('invitation_created', 'invitation_created'), ('invitation_used', 'invitation_used')])), | ||||||
|                 ('date', models.DateTimeField(auto_now_add=True)), |                 ('date', models.DateTimeField(auto_now_add=True)), | ||||||
|                 ('app', models.TextField()), |                 ('app', models.TextField()), | ||||||
|                 ('_context', models.TextField()), |                 ('context', django.contrib.postgres.fields.jsonb.JSONField(blank=True, default=dict)), | ||||||
|                 ('request_ip', models.GenericIPAddressField()), |                 ('request_ip', models.GenericIPAddressField()), | ||||||
|                 ('created', models.DateTimeField(auto_now_add=True)), |                 ('created', models.DateTimeField(auto_now_add=True)), | ||||||
|                 ('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)), |                 ('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL)), | ||||||
| @ -33,19 +34,4 @@ class Migration(migrations.Migration): | |||||||
|                 'verbose_name_plural': 'Audit Entries', |                 'verbose_name_plural': 'Audit Entries', | ||||||
|             }, |             }, | ||||||
|         ), |         ), | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='LoginAttempt', |  | ||||||
|             fields=[ |  | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |  | ||||||
|                 ('created', models.DateField(auto_now_add=True)), |  | ||||||
|                 ('last_updated', models.DateTimeField(auto_now=True)), |  | ||||||
|                 ('target_uid', models.CharField(max_length=254)), |  | ||||||
|                 ('request_ip', models.GenericIPAddressField()), |  | ||||||
|                 ('attempts', models.IntegerField(default=1)), |  | ||||||
|             ], |  | ||||||
|         ), |  | ||||||
|         migrations.AlterUniqueTogether( |  | ||||||
|             name='loginattempt', |  | ||||||
|             unique_together={('target_uid', 'request_ip', 'created')}, |  | ||||||
|         ), |  | ||||||
|     ] |     ] | ||||||
|  | |||||||
| @ -1,18 +0,0 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-21 12:01 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('passbook_audit', '0001_initial'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AlterField( |  | ||||||
|             model_name='loginattempt', |  | ||||||
|             name='created', |  | ||||||
|             field=models.DateTimeField(auto_now_add=True), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @ -1,23 +0,0 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-21 12:40 |  | ||||||
|  |  | ||||||
| import django.contrib.postgres.fields.jsonb |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('passbook_audit', '0002_auto_20190221_1201'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RemoveField( |  | ||||||
|             model_name='auditentry', |  | ||||||
|             name='_context', |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='auditentry', |  | ||||||
|             name='context', |  | ||||||
|             field=django.contrib.postgres.fields.jsonb.JSONField(blank=True, default=dict), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @ -1,19 +1,16 @@ | |||||||
| """passbook audit models""" | """passbook audit models""" | ||||||
| from datetime import timedelta |  | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.models import AnonymousUser | from django.contrib.auth.models import AnonymousUser | ||||||
| from django.contrib.postgres.fields import JSONField | from django.contrib.postgres.fields import JSONField | ||||||
| from django.core.exceptions import ValidationError | from django.core.exceptions import ValidationError | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils import timezone |  | ||||||
| from django.utils.translation import gettext as _ | from django.utils.translation import gettext as _ | ||||||
| from ipware import get_client_ip | from ipware import get_client_ip | ||||||
|  | from structlog import get_logger | ||||||
|  |  | ||||||
| from passbook.lib.models import CreatedUpdatedModel, UUIDModel | from passbook.lib.models import UUIDModel | ||||||
|  |  | ||||||
| LOGGER = getLogger(__name__) | LOGGER = get_logger() | ||||||
|  |  | ||||||
| class AuditEntry(UUIDModel): | class AuditEntry(UUIDModel): | ||||||
|     """An individual audit log entry""" |     """An individual audit log entry""" | ||||||
| @ -24,7 +21,7 @@ class AuditEntry(UUIDModel): | |||||||
|     ACTION_AUTHORIZE_APPLICATION = 'authorize_application' |     ACTION_AUTHORIZE_APPLICATION = 'authorize_application' | ||||||
|     ACTION_SUSPICIOUS_REQUEST = 'suspicious_request' |     ACTION_SUSPICIOUS_REQUEST = 'suspicious_request' | ||||||
|     ACTION_SIGN_UP = 'sign_up' |     ACTION_SIGN_UP = 'sign_up' | ||||||
|     ACTION_PASSWORD_RESET = 'password_reset' # noqa |     ACTION_PASSWORD_RESET = 'password_reset' # noqa # nosec | ||||||
|     ACTION_INVITE_CREATED = 'invitation_created' |     ACTION_INVITE_CREATED = 'invitation_created' | ||||||
|     ACTION_INVITE_USED = 'invitation_used' |     ACTION_INVITE_USED = 'invitation_used' | ||||||
|     ACTIONS = ( |     ACTIONS = ( | ||||||
| @ -75,43 +72,3 @@ class AuditEntry(UUIDModel): | |||||||
|  |  | ||||||
|         verbose_name = _('Audit Entry') |         verbose_name = _('Audit Entry') | ||||||
|         verbose_name_plural = _('Audit Entries') |         verbose_name_plural = _('Audit Entries') | ||||||
|  |  | ||||||
|  |  | ||||||
| class LoginAttempt(CreatedUpdatedModel): |  | ||||||
|     """Track failed login-attempts""" |  | ||||||
|  |  | ||||||
|     target_uid = models.CharField(max_length=254) |  | ||||||
|     request_ip = models.GenericIPAddressField() |  | ||||||
|     attempts = models.IntegerField(default=1) |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def attempt(target_uid, request): |  | ||||||
|         """Helper function to create attempt or count up existing one""" |  | ||||||
|         if not target_uid: |  | ||||||
|             return |  | ||||||
|         client_ip, _ = get_client_ip(request) |  | ||||||
|         # Since we can only use 254 chars for target_uid, truncate target_uid. |  | ||||||
|         target_uid = target_uid[:254] |  | ||||||
|         time_threshold = timezone.now() - timedelta(minutes=10) |  | ||||||
|         existing_attempts = LoginAttempt.objects.filter( |  | ||||||
|             target_uid=target_uid, |  | ||||||
|             request_ip=client_ip, |  | ||||||
|             last_updated__gt=time_threshold).order_by('created') |  | ||||||
|         if existing_attempts.exists(): |  | ||||||
|             attempt = existing_attempts.first() |  | ||||||
|             attempt.attempts += 1 |  | ||||||
|             attempt.save() |  | ||||||
|             LOGGER.debug("Increased attempts on %s", attempt) |  | ||||||
|         else: |  | ||||||
|             attempt = LoginAttempt.objects.create( |  | ||||||
|                 target_uid=target_uid, |  | ||||||
|                 request_ip=client_ip) |  | ||||||
|             LOGGER.debug("Created new attempt %s", attempt) |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return "LoginAttempt to %s from %s (x%d)" % (self.target_uid, |  | ||||||
|                                                      self.request_ip, self.attempts) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         unique_together = (('target_uid', 'request_ip', 'created'),) |  | ||||||
|  | |||||||
| @ -1 +0,0 @@ | |||||||
| django-ipware |  | ||||||
| @ -1,9 +1,8 @@ | |||||||
| """passbook audit signal listener""" | """passbook audit signal listener""" | ||||||
| from django.contrib.auth.signals import (user_logged_in, user_logged_out, | from django.contrib.auth.signals import user_logged_in, user_logged_out | ||||||
|                                          user_login_failed) |  | ||||||
| from django.dispatch import receiver | from django.dispatch import receiver | ||||||
|  |  | ||||||
| from passbook.audit.models import AuditEntry, LoginAttempt | from passbook.audit.models import AuditEntry | ||||||
| from passbook.core.signals import (invitation_created, invitation_used, | from passbook.core.signals import (invitation_created, invitation_used, | ||||||
|                                    user_signed_up) |                                    user_signed_up) | ||||||
|  |  | ||||||
| @ -34,8 +33,3 @@ def on_invitation_used(sender, request, invitation, **kwargs): | |||||||
|     """Log Invitation usage""" |     """Log Invitation usage""" | ||||||
|     AuditEntry.create(AuditEntry.ACTION_INVITE_USED, request, |     AuditEntry.create(AuditEntry.ACTION_INVITE_USED, request, | ||||||
|                       invitation_uuid=invitation.uuid.hex) |                       invitation_uuid=invitation.uuid.hex) | ||||||
|  |  | ||||||
| @receiver(user_login_failed) |  | ||||||
| def on_user_login_failed(sender, request, credentials, **kwargs): |  | ||||||
|     """Log failed login attempt""" |  | ||||||
|     LoginAttempt.attempt(target_uid=credentials.get('username'), request=request) |  | ||||||
|  | |||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook captcha_factor Header""" |  | ||||||
| __version__ = '0.0.13-alpha' |  | ||||||
| @ -1,10 +0,0 @@ | |||||||
| """passbook captcha app""" |  | ||||||
| from django.apps import AppConfig |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PassbookCaptchaFactorConfig(AppConfig): |  | ||||||
|     """passbook captcha app""" |  | ||||||
|  |  | ||||||
|     name = 'passbook.captcha_factor' |  | ||||||
|     label = 'passbook_captcha_factor' |  | ||||||
|     verbose_name = 'passbook Captcha' |  | ||||||
| @ -1,15 +0,0 @@ | |||||||
| """passbook captcha factor""" |  | ||||||
|  |  | ||||||
| from django.views.generic import FormView |  | ||||||
|  |  | ||||||
| from passbook.captcha_factor.forms import CaptchaForm |  | ||||||
| from passbook.core.auth.factor import AuthenticationFactor |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class CaptchaFactor(FormView, AuthenticationFactor): |  | ||||||
|     """Simple captcha checker, logic is handeled in django-captcha module""" |  | ||||||
|  |  | ||||||
|     form_class = CaptchaForm |  | ||||||
|  |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         return self.authenticator.user_ok() |  | ||||||
| @ -1 +0,0 @@ | |||||||
| django-recaptcha |  | ||||||
| @ -1,2 +0,0 @@ | |||||||
| """passbook core""" |  | ||||||
| __version__ = '0.0.13-alpha' |  | ||||||
|  | |||||||
| @ -1,12 +1,12 @@ | |||||||
| """passbook core app config""" | """passbook core app config""" | ||||||
| from importlib import import_module | from importlib import import_module | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|  | from django.conf import settings | ||||||
|  | from structlog import get_logger | ||||||
|  |  | ||||||
| from passbook.lib.config import CONFIG | LOGGER = get_logger() | ||||||
|  |  | ||||||
| LOGGER = getLogger(__name__) |  | ||||||
|  |  | ||||||
| class PassbookCoreConfig(AppConfig): | class PassbookCoreConfig(AppConfig): | ||||||
|     """passbook core app config""" |     """passbook core app config""" | ||||||
| @ -14,13 +14,12 @@ class PassbookCoreConfig(AppConfig): | |||||||
|     name = 'passbook.core' |     name = 'passbook.core' | ||||||
|     label = 'passbook_core' |     label = 'passbook_core' | ||||||
|     verbose_name = 'passbook Core' |     verbose_name = 'passbook Core' | ||||||
|  |     mountpoint = '' | ||||||
|  |  | ||||||
|     def ready(self): |     def ready(self): | ||||||
|         import_module('passbook.core.policies') |         for factors_to_load in settings.PASSBOOK_CORE_FACTORS: | ||||||
|         factors_to_load = CONFIG.y('passbook.factors', []) |  | ||||||
|         for factors_to_load in factors_to_load: |  | ||||||
|             try: |             try: | ||||||
|                 import_module(factors_to_load) |                 import_module(factors_to_load) | ||||||
|                 LOGGER.info("Loaded %s", factors_to_load) |                 LOGGER.info("Loaded factor", factor_class=factors_to_load) | ||||||
|             except ImportError as exc: |             except ImportError as exc: | ||||||
|                 LOGGER.debug(exc) |                 LOGGER.debug(exc) | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| """passbook Core Application forms""" | """passbook Core Application forms""" | ||||||
| from django import forms | from django import forms | ||||||
|  | from django.contrib.admin.widgets import FilteredSelectMultiple | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
| from passbook.core.models import Application, Provider | from passbook.core.models import Application, Provider | ||||||
| @ -15,11 +16,12 @@ class ApplicationForm(forms.ModelForm): | |||||||
|  |  | ||||||
|         model = Application |         model = Application | ||||||
|         fields = ['name', 'slug', 'launch_url', 'icon_url', |         fields = ['name', 'slug', 'launch_url', 'icon_url', | ||||||
|                   'policies', 'provider', 'skip_authorization'] |                   'provider', 'policies', 'skip_authorization'] | ||||||
|         widgets = { |         widgets = { | ||||||
|             'name': forms.TextInput(), |             'name': forms.TextInput(), | ||||||
|             'launch_url': forms.TextInput(), |             'launch_url': forms.TextInput(), | ||||||
|             'icon_url': forms.TextInput(), |             'icon_url': forms.TextInput(), | ||||||
|  |             'policies': FilteredSelectMultiple(_('policies'), False) | ||||||
|         } |         } | ||||||
|         labels = { |         labels = { | ||||||
|             'launch_url': _('Launch URL'), |             'launch_url': _('Launch URL'), | ||||||
|  | |||||||
| @ -1,16 +1,15 @@ | |||||||
| """passbook core authentication forms""" | """passbook core authentication forms""" | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| from django import forms | from django import forms | ||||||
| from django.core.exceptions import ValidationError | from django.core.exceptions import ValidationError | ||||||
| from django.core.validators import validate_email | from django.core.validators import validate_email | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from structlog import get_logger | ||||||
|  |  | ||||||
| from passbook.core.models import User | from passbook.core.models import User | ||||||
| from passbook.lib.config import CONFIG | from passbook.lib.config import CONFIG | ||||||
| from passbook.lib.utils.ui import human_list | from passbook.lib.utils.ui import human_list | ||||||
|  |  | ||||||
| LOGGER = getLogger(__name__) | LOGGER = get_logger() | ||||||
|  |  | ||||||
| class LoginForm(forms.Form): | class LoginForm(forms.Form): | ||||||
|     """Allow users to login""" |     """Allow users to login""" | ||||||
| @ -81,8 +80,6 @@ class SignUpForm(forms.Form): | |||||||
|         password_repeat = self.cleaned_data.get('password_repeat') |         password_repeat = self.cleaned_data.get('password_repeat') | ||||||
|         if password != password_repeat: |         if password != password_repeat: | ||||||
|             raise ValidationError(_("Passwords don't match")) |             raise ValidationError(_("Passwords don't match")) | ||||||
|         # TODO: Password policy? Via Plugin? via Policy? |  | ||||||
|         # return check_password(self) |  | ||||||
|         return self.cleaned_data.get('password_repeat') |         return self.cleaned_data.get('password_repeat') | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -91,5 +88,6 @@ class PasswordFactorForm(forms.Form): | |||||||
|  |  | ||||||
|     password = forms.CharField(widget=forms.PasswordInput(attrs={ |     password = forms.CharField(widget=forms.PasswordInput(attrs={ | ||||||
|         'placeholder': _('Password'), |         'placeholder': _('Password'), | ||||||
|         'autofocus': 'autofocus' |         'autofocus': 'autofocus', | ||||||
|  |         'autocomplete': 'current-password' | ||||||
|         })) |         })) | ||||||
|  | |||||||
| @ -1,30 +0,0 @@ | |||||||
| """passbook administration forms""" |  | ||||||
| from django import forms |  | ||||||
|  |  | ||||||
| from passbook.core.models import DummyFactor, PasswordFactor |  | ||||||
|  |  | ||||||
| GENERAL_FIELDS = ['name', 'slug', 'order', 'policies', 'enabled'] |  | ||||||
|  |  | ||||||
| class PasswordFactorForm(forms.ModelForm): |  | ||||||
|     """Form to create/edit Password Factors""" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         model = PasswordFactor |  | ||||||
|         fields = GENERAL_FIELDS + ['backends', 'password_policies'] |  | ||||||
|         widgets = { |  | ||||||
|             'name': forms.TextInput(), |  | ||||||
|             'order': forms.NumberInput(), |  | ||||||
|         } |  | ||||||
|  |  | ||||||
| class DummyFactorForm(forms.ModelForm): |  | ||||||
|     """Form to create/edit Dummy Factor""" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         model = DummyFactor |  | ||||||
|         fields = GENERAL_FIELDS |  | ||||||
|         widgets = { |  | ||||||
|             'name': forms.TextInput(), |  | ||||||
|             'order': forms.NumberInput(), |  | ||||||
|         } |  | ||||||
							
								
								
									
										32
									
								
								passbook/core/forms/groups.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								passbook/core/forms/groups.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | """passbook Core Group forms""" | ||||||
|  | from django import forms | ||||||
|  | from django.contrib.admin.widgets import FilteredSelectMultiple | ||||||
|  |  | ||||||
|  | from passbook.core.models import Group, User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GroupForm(forms.ModelForm): | ||||||
|  |     """Group Form""" | ||||||
|  |  | ||||||
|  |     members = forms.ModelMultipleChoiceField( | ||||||
|  |         User.objects.all(), required=False, widget=FilteredSelectMultiple('users', False)) | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |         if self.instance.pk: | ||||||
|  |             self.initial['members'] = self.instance.user_set.values_list('pk', flat=True) | ||||||
|  |  | ||||||
|  |     def save(self, *args, **kwargs): | ||||||
|  |         instance = super().save(*args, **kwargs) | ||||||
|  |         if instance.pk: | ||||||
|  |             instance.user_set.clear() | ||||||
|  |             instance.user_set.add(*self.cleaned_data['members']) | ||||||
|  |         return instance | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         model = Group | ||||||
|  |         fields = ['name', 'parent', 'members', 'tags'] | ||||||
|  |         widgets = { | ||||||
|  |             'name': forms.TextInput(), | ||||||
|  |         } | ||||||
| @ -3,39 +3,8 @@ | |||||||
| from django import forms | from django import forms | ||||||
| from django.utils.translation import gettext as _ | from django.utils.translation import gettext as _ | ||||||
|  |  | ||||||
| from passbook.core.models import (DebugPolicy, FieldMatcherPolicy, | from passbook.core.models import DebugPolicy | ||||||
|                                   PasswordPolicy, WebhookPolicy) | from passbook.policies.forms import GENERAL_FIELDS | ||||||
|  |  | ||||||
| GENERAL_FIELDS = ['name', 'action', 'negate', 'order', ] |  | ||||||
|  |  | ||||||
| class FieldMatcherPolicyForm(forms.ModelForm): |  | ||||||
|     """FieldMatcherPolicy Form""" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         model = FieldMatcherPolicy |  | ||||||
|         fields = GENERAL_FIELDS + ['user_field', 'match_action', 'value', ] |  | ||||||
|         widgets = { |  | ||||||
|             'name': forms.TextInput(), |  | ||||||
|             'value': forms.TextInput(), |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WebhookPolicyForm(forms.ModelForm): |  | ||||||
|     """WebhookPolicyForm Form""" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         model = WebhookPolicy |  | ||||||
|         fields = GENERAL_FIELDS + ['url', 'method', 'json_body', 'json_headers', |  | ||||||
|                                    'result_jsonpath', 'result_json_value', ] |  | ||||||
|         widgets = { |  | ||||||
|             'name': forms.TextInput(), |  | ||||||
|             'json_body': forms.TextInput(), |  | ||||||
|             'json_headers': forms.TextInput(), |  | ||||||
|             'result_jsonpath': forms.TextInput(), |  | ||||||
|             'result_json_value': forms.TextInput(), |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DebugPolicyForm(forms.ModelForm): | class DebugPolicyForm(forms.ModelForm): | ||||||
| @ -51,25 +20,3 @@ class DebugPolicyForm(forms.ModelForm): | |||||||
|         labels = { |         labels = { | ||||||
|             'result': _('Allow user') |             'result': _('Allow user') | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class PasswordPolicyForm(forms.ModelForm): |  | ||||||
|     """PasswordPolicy Form""" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|  |  | ||||||
|         model = PasswordPolicy |  | ||||||
|         fields = GENERAL_FIELDS + ['amount_uppercase', 'amount_lowercase', |  | ||||||
|                                    'amount_symbols', 'length_min', 'symbol_charset', |  | ||||||
|                                    'error_message'] |  | ||||||
|         widgets = { |  | ||||||
|             'name': forms.TextInput(), |  | ||||||
|             'symbol_charset': forms.TextInput(), |  | ||||||
|             'error_message': forms.TextInput(), |  | ||||||
|         } |  | ||||||
|         labels = { |  | ||||||
|             'amount_uppercase': _('Minimum amount of Uppercase Characters'), |  | ||||||
|             'amount_lowercase': _('Minimum amount of Lowercase Characters'), |  | ||||||
|             'amount_symbols': _('Minimum amount of Symbols Characters'), |  | ||||||
|             'length_min': _('Minimum Length'), |  | ||||||
|         } |  | ||||||
|  | |||||||
| @ -22,10 +22,14 @@ class PasswordChangeForm(forms.Form): | |||||||
|     """Form to update password""" |     """Form to update password""" | ||||||
|  |  | ||||||
|     password = forms.CharField(label=_('Password'), |     password = forms.CharField(label=_('Password'), | ||||||
|                                widget=forms.PasswordInput(attrs={'placeholder': _('New Password')})) |                                widget=forms.PasswordInput(attrs={ | ||||||
|  |                                    'placeholder': _('New Password'), | ||||||
|  |                                    'autocomplete': 'new-password' | ||||||
|  |                                    })) | ||||||
|     password_repeat = forms.CharField(label=_('Repeat Password'), |     password_repeat = forms.CharField(label=_('Repeat Password'), | ||||||
|                                       widget=forms.PasswordInput(attrs={ |                                       widget=forms.PasswordInput(attrs={ | ||||||
|                                           'placeholder': _('Repeat Password') |                                           'placeholder': _('Repeat Password'), | ||||||
|  |                                           'autocomplete': 'new-password' | ||||||
|                                       })) |                                       })) | ||||||
|  |  | ||||||
|     def clean_password_repeat(self): |     def clean_password_repeat(self): | ||||||
| @ -34,5 +38,4 @@ class PasswordChangeForm(forms.Form): | |||||||
|         password_repeat = self.cleaned_data.get('password_repeat') |         password_repeat = self.cleaned_data.get('password_repeat') | ||||||
|         if password != password_repeat: |         if password != password_repeat: | ||||||
|             raise ValidationError(_("Passwords don't match")) |             raise ValidationError(_("Passwords don't match")) | ||||||
|         # TODO: Password policy check |  | ||||||
|         return self.cleaned_data.get('password_repeat') |         return self.cleaned_data.get('password_repeat') | ||||||
|  | |||||||
| @ -1,63 +0,0 @@ | |||||||
| """passbook nexus_upload management command""" |  | ||||||
| from base64 import b64decode |  | ||||||
|  |  | ||||||
| import requests |  | ||||||
| from django.core.management.base import BaseCommand |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Command(BaseCommand): |  | ||||||
|     """Upload debian package to nexus repository""" |  | ||||||
|  |  | ||||||
|     url = None |  | ||||||
|     user = None |  | ||||||
|     password = None |  | ||||||
|  |  | ||||||
|     def add_arguments(self, parser): |  | ||||||
|         parser.add_argument( |  | ||||||
|             '--repo', |  | ||||||
|             action='store', |  | ||||||
|             help='Repository to upload to', |  | ||||||
|             required=True) |  | ||||||
|         parser.add_argument( |  | ||||||
|             '--url', |  | ||||||
|             action='store', |  | ||||||
|             help='Nexus root URL', |  | ||||||
|             required=True) |  | ||||||
|         parser.add_argument( |  | ||||||
|             '--auth', |  | ||||||
|             action='store', |  | ||||||
|             help='base64-encoded string of username:password', |  | ||||||
|             required=True) |  | ||||||
|         parser.add_argument( |  | ||||||
|             '--method', |  | ||||||
|             action='store', |  | ||||||
|             nargs='?', |  | ||||||
|             const='post', |  | ||||||
|             choices=['post', 'put'], |  | ||||||
|             help=('Method used for uploading files to nexus. ' |  | ||||||
|                   'Apt repositories use post, Helm uses put.'), |  | ||||||
|             required=True) |  | ||||||
|         # Positional arguments |  | ||||||
|         parser.add_argument('file', nargs='+', type=str) |  | ||||||
|  |  | ||||||
|     def handle(self, *args, **options): |  | ||||||
|         """Upload debian package to nexus repository""" |  | ||||||
|         auth = tuple(b64decode(options.get('auth')).decode('utf-8').split(':', 1)) |  | ||||||
|         responses = {} |  | ||||||
|         url = 'https://%(url)s/repository/%(repo)s/' % options |  | ||||||
|         method = options.get('method') |  | ||||||
|         exit_code = 0 |  | ||||||
|         for file in options.get('file'): |  | ||||||
|             if method == 'post': |  | ||||||
|                 responses[file] = requests.post(url, data=open(file, mode='rb'), auth=auth) |  | ||||||
|             else: |  | ||||||
|                 responses[file] = requests.put(url+file, data=open(file, mode='rb'), auth=auth) |  | ||||||
|         self.stdout.write('Upload results:\n') |  | ||||||
|         sep = '-' * 60 |  | ||||||
|         self.stdout.write('%s\n' % sep) |  | ||||||
|         for path, response in responses.items(): |  | ||||||
|             self.stdout.write('%-55s: %d\n' % (path, response.status_code)) |  | ||||||
|             if response.status_code >= 400: |  | ||||||
|                 exit_code = 1 |  | ||||||
|         self.stdout.write('%s\n' % sep) |  | ||||||
|         exit(exit_code) |  | ||||||
| @ -1,34 +0,0 @@ | |||||||
| """passbook Webserver management command""" |  | ||||||
|  |  | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| import cherrypy |  | ||||||
| from django.conf import settings |  | ||||||
| from django.core.management.base import BaseCommand |  | ||||||
|  |  | ||||||
| from passbook.core.wsgi import application |  | ||||||
|  |  | ||||||
| LOGGER = getLogger(__name__) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Command(BaseCommand): |  | ||||||
|     """Run CherryPy webserver""" |  | ||||||
|  |  | ||||||
|     def handle(self, *args, **options): |  | ||||||
|         """passbook cherrypy server""" |  | ||||||
|         config = settings.CHERRYPY_SERVER |  | ||||||
|         config.update(**options) |  | ||||||
|         cherrypy.config.update(config) |  | ||||||
|         cherrypy.tree.graft(application, '/') |  | ||||||
|         # Mount NullObject to serve static files |  | ||||||
|         cherrypy.tree.mount(None, '/static', config={ |  | ||||||
|             '/': { |  | ||||||
|                 'tools.staticdir.on': True, |  | ||||||
|                 'tools.staticdir.dir': settings.STATIC_ROOT, |  | ||||||
|                 'tools.expires.on': True, |  | ||||||
|                 'tools.expires.secs': 86400, |  | ||||||
|                 'tools.gzip.on': True, |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|         cherrypy.engine.start() |  | ||||||
|         cherrypy.engine.block() |  | ||||||
| @ -1,17 +0,0 @@ | |||||||
| """passbook Worker management command""" |  | ||||||
|  |  | ||||||
| from logging import getLogger |  | ||||||
|  |  | ||||||
| from django.core.management.base import BaseCommand |  | ||||||
|  |  | ||||||
| from passbook.core.celery import CELERY_APP |  | ||||||
|  |  | ||||||
| LOGGER = getLogger(__name__) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Command(BaseCommand): |  | ||||||
|     """Run Celery Worker""" |  | ||||||
|  |  | ||||||
|     def handle(self, *args, **options): |  | ||||||
|         """celery worker""" |  | ||||||
|         CELERY_APP.worker_main(['worker', '--autoscale=10,3', '-E']) |  | ||||||
| @ -1,21 +1,24 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-16 09:10 | # Generated by Django 2.2.6 on 2019-10-07 14:06 | ||||||
|  |  | ||||||
| import uuid | import uuid | ||||||
|  |  | ||||||
| import django.contrib.auth.models | import django.contrib.auth.models | ||||||
| import django.contrib.auth.validators | import django.contrib.auth.validators | ||||||
|  | import django.contrib.postgres.fields.jsonb | ||||||
| import django.db.models.deletion | import django.db.models.deletion | ||||||
| import django.utils.timezone | import django.utils.timezone | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db import migrations, models | from django.db import migrations, models | ||||||
|  |  | ||||||
|  | import passbook.core.models | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|     initial = True |     initial = True | ||||||
|  |  | ||||||
|     dependencies = [ |     dependencies = [ | ||||||
|         ('auth', '0009_alter_user_last_name_max_length'), |         ('auth', '0011_update_proxy_permissions'), | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|     operations = [ |     operations = [ | ||||||
| @ -34,6 +37,8 @@ class Migration(migrations.Migration): | |||||||
|                 ('is_active', models.BooleanField(default=True, help_text='Designates whether this user should be treated as active. Unselect this instead of deleting accounts.', verbose_name='active')), |                 ('is_active', models.BooleanField(default=True, help_text='Designates whether this user should be treated as active. Unselect this instead of deleting accounts.', verbose_name='active')), | ||||||
|                 ('date_joined', models.DateTimeField(default=django.utils.timezone.now, verbose_name='date joined')), |                 ('date_joined', models.DateTimeField(default=django.utils.timezone.now, verbose_name='date joined')), | ||||||
|                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False)), |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False)), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |                 ('password_change_date', models.DateTimeField(auto_now_add=True)), | ||||||
|             ], |             ], | ||||||
|             options={ |             options={ | ||||||
|                 'verbose_name': 'user', |                 'verbose_name': 'user', | ||||||
| @ -44,39 +49,17 @@ class Migration(migrations.Migration): | |||||||
|                 ('objects', django.contrib.auth.models.UserManager()), |                 ('objects', django.contrib.auth.models.UserManager()), | ||||||
|             ], |             ], | ||||||
|         ), |         ), | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Group', |  | ||||||
|             fields=[ |  | ||||||
|                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), |  | ||||||
|                 ('name', models.CharField(max_length=80, verbose_name='name')), |  | ||||||
|                 ('extra_data', models.TextField(blank=True)), |  | ||||||
|                 ('parent', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='children', to='passbook_core.Group')), |  | ||||||
|             ], |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Invitation', |  | ||||||
|             fields=[ |  | ||||||
|                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), |  | ||||||
|                 ('expires', models.DateTimeField(blank=True, default=None, null=True)), |  | ||||||
|                 ('fixed_username', models.TextField(blank=True, default=None)), |  | ||||||
|                 ('fixed_email', models.TextField(blank=True, default=None)), |  | ||||||
|                 ('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Invitation', |  | ||||||
|                 'verbose_name_plural': 'Invitations', |  | ||||||
|             }, |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |         migrations.CreateModel( | ||||||
|             name='Policy', |             name='Policy', | ||||||
|             fields=[ |             fields=[ | ||||||
|                 ('created', models.DateField(auto_now_add=True)), |                 ('created', models.DateTimeField(auto_now_add=True)), | ||||||
|                 ('last_updated', models.DateTimeField(auto_now=True)), |                 ('last_updated', models.DateTimeField(auto_now=True)), | ||||||
|                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|                 ('name', models.TextField(blank=True, null=True)), |                 ('name', models.TextField(blank=True, null=True)), | ||||||
|                 ('action', models.CharField(choices=[('allow', 'allow'), ('deny', 'deny')], max_length=20)), |                 ('action', models.CharField(choices=[('allow', 'allow'), ('deny', 'deny')], max_length=20)), | ||||||
|                 ('negate', models.BooleanField(default=False)), |                 ('negate', models.BooleanField(default=False)), | ||||||
|                 ('order', models.IntegerField(default=0)), |                 ('order', models.IntegerField(default=0)), | ||||||
|  |                 ('timeout', models.IntegerField(default=30)), | ||||||
|             ], |             ], | ||||||
|             options={ |             options={ | ||||||
|                 'abstract': False, |                 'abstract': False, | ||||||
| @ -85,28 +68,136 @@ class Migration(migrations.Migration): | |||||||
|         migrations.CreateModel( |         migrations.CreateModel( | ||||||
|             name='PolicyModel', |             name='PolicyModel', | ||||||
|             fields=[ |             fields=[ | ||||||
|                 ('created', models.DateField(auto_now_add=True)), |                 ('created', models.DateTimeField(auto_now_add=True)), | ||||||
|                 ('last_updated', models.DateTimeField(auto_now=True)), |                 ('last_updated', models.DateTimeField(auto_now=True)), | ||||||
|                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|  |                 ('policies', models.ManyToManyField(blank=True, to='passbook_core.Policy')), | ||||||
|             ], |             ], | ||||||
|             options={ |             options={ | ||||||
|                 'abstract': False, |                 'abstract': False, | ||||||
|             }, |             }, | ||||||
|         ), |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='PropertyMapping', | ||||||
|  |             fields=[ | ||||||
|  |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Property Mapping', | ||||||
|  |                 'verbose_name_plural': 'Property Mappings', | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='DebugPolicy', | ||||||
|  |             fields=[ | ||||||
|  |                 ('policy_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Policy')), | ||||||
|  |                 ('result', models.BooleanField(default=False)), | ||||||
|  |                 ('wait_min', models.IntegerField(default=5)), | ||||||
|  |                 ('wait_max', models.IntegerField(default=30)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Debug Policy', | ||||||
|  |                 'verbose_name_plural': 'Debug Policies', | ||||||
|  |             }, | ||||||
|  |             bases=('passbook_core.policy',), | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Factor', | ||||||
|  |             fields=[ | ||||||
|  |                 ('policymodel_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.PolicyModel')), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |                 ('slug', models.SlugField(unique=True)), | ||||||
|  |                 ('order', models.IntegerField()), | ||||||
|  |                 ('enabled', models.BooleanField(default=True)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'abstract': False, | ||||||
|  |             }, | ||||||
|  |             bases=('passbook_core.policymodel',), | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Source', | ||||||
|  |             fields=[ | ||||||
|  |                 ('policymodel_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.PolicyModel')), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |                 ('slug', models.SlugField()), | ||||||
|  |                 ('enabled', models.BooleanField(default=True)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'abstract': False, | ||||||
|  |             }, | ||||||
|  |             bases=('passbook_core.policymodel',), | ||||||
|  |         ), | ||||||
|         migrations.CreateModel( |         migrations.CreateModel( | ||||||
|             name='Provider', |             name='Provider', | ||||||
|             fields=[ |             fields=[ | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('property_mappings', models.ManyToManyField(blank=True, default=None, to='passbook_core.PropertyMapping')), | ||||||
|             ], |             ], | ||||||
|         ), |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Nonce', | ||||||
|  |             fields=[ | ||||||
|  |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|  |                 ('expires', models.DateTimeField(default=passbook.core.models.default_nonce_duration)), | ||||||
|  |                 ('expiring', models.BooleanField(default=True)), | ||||||
|  |                 ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Nonce', | ||||||
|  |                 'verbose_name_plural': 'Nonces', | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Invitation', | ||||||
|  |             fields=[ | ||||||
|  |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|  |                 ('expires', models.DateTimeField(blank=True, default=None, null=True)), | ||||||
|  |                 ('fixed_username', models.TextField(blank=True, default=None)), | ||||||
|  |                 ('fixed_email', models.TextField(blank=True, default=None)), | ||||||
|  |                 ('needs_confirmation', models.BooleanField(default=True)), | ||||||
|  |                 ('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Invitation', | ||||||
|  |                 'verbose_name_plural': 'Invitations', | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Group', | ||||||
|  |             fields=[ | ||||||
|  |                 ('uuid', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), | ||||||
|  |                 ('name', models.CharField(max_length=80, verbose_name='name')), | ||||||
|  |                 ('tags', django.contrib.postgres.fields.jsonb.JSONField(blank=True, default=dict)), | ||||||
|  |                 ('parent', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='children', to='passbook_core.Group')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'unique_together': {('name', 'parent')}, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.AddField( | ||||||
|  |             model_name='user', | ||||||
|  |             name='groups', | ||||||
|  |             field=models.ManyToManyField(to='passbook_core.Group'), | ||||||
|  |         ), | ||||||
|  |         migrations.AddField( | ||||||
|  |             model_name='user', | ||||||
|  |             name='user_permissions', | ||||||
|  |             field=models.ManyToManyField(blank=True, help_text='Specific permissions for this user.', related_name='user_set', related_query_name='user', to='auth.Permission', verbose_name='user permissions'), | ||||||
|  |         ), | ||||||
|         migrations.CreateModel( |         migrations.CreateModel( | ||||||
|             name='UserSourceConnection', |             name='UserSourceConnection', | ||||||
|             fields=[ |             fields=[ | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|                 ('created', models.DateField(auto_now_add=True)), |                 ('created', models.DateTimeField(auto_now_add=True)), | ||||||
|                 ('last_updated', models.DateTimeField(auto_now=True)), |                 ('last_updated', models.DateTimeField(auto_now=True)), | ||||||
|                 ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), |                 ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), | ||||||
|  |                 ('source', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='passbook_core.Source')), | ||||||
|             ], |             ], | ||||||
|  |             options={ | ||||||
|  |                 'unique_together': {('user', 'source')}, | ||||||
|  |             }, | ||||||
|         ), |         ), | ||||||
|         migrations.CreateModel( |         migrations.CreateModel( | ||||||
|             name='Application', |             name='Application', | ||||||
| @ -124,131 +215,9 @@ class Migration(migrations.Migration): | |||||||
|             }, |             }, | ||||||
|             bases=('passbook_core.policymodel',), |             bases=('passbook_core.policymodel',), | ||||||
|         ), |         ), | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='DebugPolicy', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policy_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Policy')), |  | ||||||
|                 ('result', models.BooleanField(default=False)), |  | ||||||
|                 ('wait_min', models.IntegerField(default=5)), |  | ||||||
|                 ('wait_max', models.IntegerField(default=30)), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Debug Policy', |  | ||||||
|                 'verbose_name_plural': 'Debug Policys', |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policy',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Factor', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policymodel_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.PolicyModel')), |  | ||||||
|                 ('name', models.TextField()), |  | ||||||
|                 ('slug', models.SlugField(unique=True)), |  | ||||||
|                 ('order', models.IntegerField()), |  | ||||||
|                 ('type', models.TextField(unique=True)), |  | ||||||
|                 ('enabled', models.BooleanField(default=True)), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'abstract': False, |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policymodel',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='FieldMatcherPolicy', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policy_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Policy')), |  | ||||||
|                 ('user_field', models.TextField(choices=[('username', 'Username'), ('first_name', 'First Name'), ('last_name', 'Last Name'), ('email', 'E-Mail'), ('is_staff', 'Is staff'), ('is_active', 'Is active'), ('data_joined', 'Date joined')])), |  | ||||||
|                 ('match_action', models.CharField(choices=[('startswith', 'Starts with'), ('endswith', 'Ends with'), ('endswith', 'Contains'), ('regexp', 'Regexp'), ('exact', 'Exact')], max_length=50)), |  | ||||||
|                 ('value', models.TextField()), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Field matcher Policy', |  | ||||||
|                 'verbose_name_plural': 'Field matcher Policys', |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policy',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='PasswordPolicyPolicy', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policy_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Policy')), |  | ||||||
|                 ('amount_uppercase', models.IntegerField(default=0)), |  | ||||||
|                 ('amount_lowercase', models.IntegerField(default=0)), |  | ||||||
|                 ('amount_symbols', models.IntegerField(default=0)), |  | ||||||
|                 ('length_min', models.IntegerField(default=0)), |  | ||||||
|                 ('symbol_charset', models.TextField(default='!\\"#$%&\'()*+,-./:;<=>?@[\\]^_`{|}~ ')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Password Policy Policy', |  | ||||||
|                 'verbose_name_plural': 'Password Policy Policys', |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policy',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Source', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policymodel_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.PolicyModel')), |  | ||||||
|                 ('name', models.TextField()), |  | ||||||
|                 ('slug', models.SlugField()), |  | ||||||
|                 ('enabled', models.BooleanField(default=True)), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'abstract': False, |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policymodel',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='WebhookPolicy', |  | ||||||
|             fields=[ |  | ||||||
|                 ('policy_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Policy')), |  | ||||||
|                 ('url', models.URLField()), |  | ||||||
|                 ('method', models.CharField(choices=[('GET', 'GET'), ('POST', 'POST'), ('PATCH', 'PATCH'), ('DELETE', 'DELETE'), ('PUT', 'PUT')], max_length=10)), |  | ||||||
|                 ('json_body', models.TextField()), |  | ||||||
|                 ('json_headers', models.TextField()), |  | ||||||
|                 ('result_jsonpath', models.TextField()), |  | ||||||
|                 ('result_json_value', models.TextField()), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Webhook Policy', |  | ||||||
|                 'verbose_name_plural': 'Webhook Policys', |  | ||||||
|             }, |  | ||||||
|             bases=('passbook_core.policy',), |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='policymodel', |  | ||||||
|             name='policies', |  | ||||||
|             field=models.ManyToManyField(blank=True, to='passbook_core.Policy'), |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='user', |  | ||||||
|             name='groups', |  | ||||||
|             field=models.ManyToManyField(to='passbook_core.Group'), |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='user', |  | ||||||
|             name='user_permissions', |  | ||||||
|             field=models.ManyToManyField(blank=True, help_text='Specific permissions for this user.', related_name='user_set', related_query_name='user', to='auth.Permission', verbose_name='user permissions'), |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='usersourceconnection', |  | ||||||
|             name='source', |  | ||||||
|             field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='passbook_core.Source'), |  | ||||||
|         ), |  | ||||||
|         migrations.AlterUniqueTogether( |  | ||||||
|             name='group', |  | ||||||
|             unique_together={('name', 'parent')}, |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='user', |  | ||||||
|             name='applications', |  | ||||||
|             field=models.ManyToManyField(to='passbook_core.Application'), |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |         migrations.AddField( | ||||||
|             model_name='user', |             model_name='user', | ||||||
|             name='sources', |             name='sources', | ||||||
|             field=models.ManyToManyField(through='passbook_core.UserSourceConnection', to='passbook_core.Source'), |             field=models.ManyToManyField(through='passbook_core.UserSourceConnection', to='passbook_core.Source'), | ||||||
|         ), |         ), | ||||||
|         migrations.AlterUniqueTogether( |  | ||||||
|             name='usersourceconnection', |  | ||||||
|             unique_together={('user', 'source')}, |  | ||||||
|         ), |  | ||||||
|     ] |     ] | ||||||
|  | |||||||
| @ -1,29 +0,0 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-16 10:02 |  | ||||||
|  |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('passbook_core', '0001_initial'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='debugpolicy', |  | ||||||
|             options={'verbose_name': 'Debug Policy', 'verbose_name_plural': 'Debug Policies'}, |  | ||||||
|         ), |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='fieldmatcherpolicy', |  | ||||||
|             options={'verbose_name': 'Field matcher Policy', 'verbose_name_plural': 'Field matcher Policies'}, |  | ||||||
|         ), |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='passwordpolicypolicy', |  | ||||||
|             options={'verbose_name': 'Password Policy Policy', 'verbose_name_plural': 'Password Policy Policies'}, |  | ||||||
|         ), |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='webhookpolicy', |  | ||||||
|             options={'verbose_name': 'Webhook Policy', 'verbose_name_plural': 'Webhook Policies'}, |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @ -1,17 +0,0 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-16 10:04 |  | ||||||
|  |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('passbook_core', '0002_auto_20190216_1002'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RenameModel( |  | ||||||
|             old_name='PasswordPolicyPolicy', |  | ||||||
|             new_name='PasswordPolicy', |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @ -1,17 +0,0 @@ | |||||||
| # Generated by Django 2.1.7 on 2019-02-16 10:13 |  | ||||||
|  |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('passbook_core', '0003_auto_20190216_1004'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='passwordpolicy', |  | ||||||
|             options={'verbose_name': 'Password Policy', 'verbose_name_plural': 'Password Policies'}, |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	