Compare commits
	
		
			199 Commits
		
	
	
		
			version/0.
			...
			version/0.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 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 | |||
| 2d7e8f1b50 | 
| @ -1,10 +1,10 @@ | |||||||
| [bumpversion] | [bumpversion] | ||||||
| current_version = 0.1.8-beta | current_version = 0.2.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] | ||||||
| @ -15,6 +15,10 @@ values = | |||||||
| 	beta | 	beta | ||||||
| 	stable | 	stable | ||||||
|  |  | ||||||
|  | [bumpversion:file:client-packages/allauth/setup.py] | ||||||
|  |  | ||||||
|  | [bumpversion:file:client-packages/sentry-auth-passbook/setup.py] | ||||||
|  |  | ||||||
| [bumpversion:file:helm/passbook/values.yaml] | [bumpversion:file:helm/passbook/values.yaml] | ||||||
|  |  | ||||||
| [bumpversion:file:helm/passbook/Chart.yaml] | [bumpversion:file:helm/passbook/Chart.yaml] | ||||||
| @ -49,3 +53,7 @@ values = | |||||||
|  |  | ||||||
| [bumpversion:file:passbook/otp/__init__.py] | [bumpversion:file:passbook/otp/__init__.py] | ||||||
|  |  | ||||||
|  | [bumpversion:file:passbook/app_gw/__init__.py] | ||||||
|  |  | ||||||
|  | [bumpversion:file:passbook/suspicious_policy/__init__.py] | ||||||
|  |  | ||||||
|  | |||||||
| @ -6,7 +6,6 @@ omit = | |||||||
|     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/ | ||||||
|  | |||||||
| @ -9,3 +9,6 @@ insert_final_newline = true | |||||||
|  |  | ||||||
| [html] | [html] | ||||||
| indent_size = 2 | indent_size = 2 | ||||||
|  |  | ||||||
|  | [yaml] | ||||||
|  | indent_size = 2 | ||||||
|  | |||||||
							
								
								
									
										100
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								.gitlab-ci.yml
									
									
									
									
									
								
							| @ -1,24 +1,33 @@ | |||||||
| # 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-buildimage | ||||||
|   - test |   - test | ||||||
|   - build |   - build | ||||||
|   - docs |   - docs | ||||||
| image: python:3.6 |   - deploy | ||||||
|  | image: docker.beryju.org/passbook/build-base:latest | ||||||
| services: | services: | ||||||
|   - postgres:latest |   - postgres:latest | ||||||
|  |   - redis: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" | ||||||
|  |  | ||||||
| include: | create-build-image: | ||||||
|   - /allauth/.gitlab-ci.yml |   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/Dockerfile.build-base --destination docker.beryju.org/passbook/build-base:latest --destination docker.beryju.org/passbook/build-base:0.2.0-beta | ||||||
|  |   stage: build-buildimage | ||||||
|  |   only: | ||||||
|  |     refs: | ||||||
|  |       - tags | ||||||
|  |       - /^version/.*$/ | ||||||
|  |  | ||||||
| isort: | isort: | ||||||
|   script: |   script: | ||||||
| @ -38,6 +47,7 @@ pylint: | |||||||
|   stage: test |   stage: test | ||||||
| coverage: | coverage: | ||||||
|   script: |   script: | ||||||
|  |     - python manage.py collectstatic --no-input | ||||||
|     - coverage run manage.py test |     - coverage run manage.py test | ||||||
|     - coverage report |     - coverage report | ||||||
|   stage: test |   stage: test | ||||||
| @ -51,9 +61,9 @@ package-docker: | |||||||
|     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.1.8-beta |     - /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.2.0-beta | ||||||
|   stage: build |   stage: build | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
| @ -64,52 +74,36 @@ package-helm: | |||||||
|     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash |     - curl https://raw.githubusercontent.com/helm/helm/master/scripts/get | bash | ||||||
|     - helm init --client-only |     - helm init --client-only | ||||||
|     - helm package helm/passbook |     - helm package helm/passbook | ||||||
|     - ./manage.py nexus_upload --method put --url $NEXUS_URL --auth $NEXUS_AUTH --repo helm *.tgz |  | ||||||
|   only: |  | ||||||
|     - tags |  | ||||||
|     - /^version/.*$/ |  | ||||||
| package-debian: |  | ||||||
|   before_script: |  | ||||||
|     - apt update |  | ||||||
|     - apt install -y --no-install-recommends build-essential debhelper devscripts equivs python3 python3-dev python3-pip libsasl2-dev libldap2-dev |  | ||||||
|     - mk-build-deps debian/control |  | ||||||
|     - apt install ./*build-deps*deb -f -y |  | ||||||
|     - python3 -m pip install -U virtualenv pip |  | ||||||
|     - virtualenv env |  | ||||||
|     - source env/bin/activate |  | ||||||
|     - pip3 install -U -r requirements.txt -r requirements-dev.txt |  | ||||||
|     - ./manage.py collectstatic --no-input |  | ||||||
|   image: ubuntu:18.04 |  | ||||||
|   script: |  | ||||||
|     - debuild -us -uc |  | ||||||
|     - cp ../passbook*.deb . |  | ||||||
|     - ./manage.py nexus_upload --method post --url $NEXUS_URL --auth $NEXUS_AUTH --repo apt passbook*deb |  | ||||||
|   artifacts: |   artifacts: | ||||||
|     paths: |     paths: | ||||||
|     - passbook*deb |       - passbook-*.tgz | ||||||
|     expire_in: 2 days |     expire_in: 2 days | ||||||
|   stage: build |  | ||||||
|   only: |   only: | ||||||
|     - tags |     - tags | ||||||
|     - /^version/.*$/ |     - /^version/.*$/ | ||||||
|  |  | ||||||
| # docs: | package-client-package-allauth: | ||||||
| #   stage: docs |   script: | ||||||
| #   only: |     - cd client-packages/allauth | ||||||
| #   - master |     - python setup.py sdist | ||||||
| #   - tags |     - twine upload --username $TWINE_USERNAME --password $TWINE_PASSWORD dist/* | ||||||
| #   - /^debian/.*$/ |   stage: build | ||||||
| #   environment: |   only: | ||||||
| #     name: docs |     refs: | ||||||
| #     url: "https://passbook.beryju.org/docs/" |       - tags | ||||||
| #   script: |       - /^version/.*$/ | ||||||
| #     - apt update |     changes: | ||||||
| #     - apt install -y rsync |       - client-packages/allauth/** | ||||||
| #     - "mkdir ~/.ssh" |  | ||||||
| #     - "cp .gitlab/known_hosts ~/.ssh/" | package-client-package-sentry: | ||||||
| #     - "pip3 install -U -r requirements-docs.txt" |   script: | ||||||
| #     - "eval $(ssh-agent -s)" |     - cd client-packages/sentry-auth-passbook | ||||||
| #     - "echo \"${CI_SSH_PRIVATE}\" | ssh-add -" |     - python setup.py sdist | ||||||
| #     - mkdocs build |     - twine upload --username $TWINE_USERNAME --password $TWINE_PASSWORD dist/* | ||||||
| #     - 'rsync -avh --delete web/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/"' |   stage: build | ||||||
| #     - 'rsync -avh --delete site/* "beryjuorg@ory1-web-prod-1.ory1.beryju.org:passbook.beryju.org/docs/"' |   only: | ||||||
|  |     refs: | ||||||
|  |       - tags | ||||||
|  |       - /^version/.*$/ | ||||||
|  |     changes: | ||||||
|  |       - client-packages/sentry-auth-passbook/** | ||||||
|  | |||||||
| @ -7,6 +7,7 @@ ignore-paths: | |||||||
|   - migrations |   - migrations | ||||||
|   - docs |   - docs | ||||||
|   - node_modules |   - node_modules | ||||||
|  |   - client-packages | ||||||
|  |  | ||||||
| uses: | uses: | ||||||
|  - django |  - django | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @ -4,6 +4,9 @@ | |||||||
|   "[html]": { |   "[html]": { | ||||||
|     "editor.tabSize": 2 |     "editor.tabSize": 2 | ||||||
|   }, |   }, | ||||||
|  |   "[yml]": { | ||||||
|  |     "editor.tabSize": 2 | ||||||
|  |   }, | ||||||
|   "cSpell.words": [ |   "cSpell.words": [ | ||||||
|     "SAML", |     "SAML", | ||||||
|     "passbook" |     "passbook" | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ COPY ./requirements.txt /app/ | |||||||
|  |  | ||||||
| WORKDIR /app/ | WORKDIR /app/ | ||||||
|  |  | ||||||
| RUN apt-get update && apt-get install build-essential libssl-dev libffi-dev -y && \ | RUN apt-get update && apt-get install build-essential libssl-dev libffi-dev libpq-dev -y && \ | ||||||
|     mkdir /app/static/ && \ |     mkdir /app/static/ && \ | ||||||
|     pip install -r requirements.txt && \ |     pip install -r requirements.txt && \ | ||||||
|     pip install psycopg2 && \ |     pip install psycopg2 && \ | ||||||
| @ -23,7 +23,7 @@ COPY --from=build /app/static /app/static/ | |||||||
|  |  | ||||||
| WORKDIR /app/ | WORKDIR /app/ | ||||||
|  |  | ||||||
| RUN apt-get update && apt-get install build-essential libssl-dev libffi-dev -y && \ | RUN apt-get update && apt-get install build-essential libssl-dev libffi-dev libpq-dev -y && \ | ||||||
|     pip install -r requirements.txt && \ |     pip install -r requirements.txt && \ | ||||||
|     pip install psycopg2 && \ |     pip install psycopg2 && \ | ||||||
|     adduser --system --home /app/ passbook && \ |     adduser --system --home /app/ passbook && \ | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								Dockerfile.build-base
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								Dockerfile.build-base
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | FROM python:3.6 | ||||||
|  |  | ||||||
|  | COPY ./passbook/ /app/passbook | ||||||
|  | COPY ./client-packages/ /app/client-packages | ||||||
|  | COPY ./requirements.txt /app/ | ||||||
|  | COPY ./requirements-dev.txt /app/ | ||||||
|  |  | ||||||
|  | WORKDIR /app/ | ||||||
|  |  | ||||||
|  | RUN apt-get update && apt-get install libssl-dev libffi-dev libpq-dev -y && \ | ||||||
|  |      pip install -U -r requirements-dev.txt && \ | ||||||
|  |      rm -rf /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 | ||||||
|  | |||||||
| @ -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,5 +1,6 @@ | |||||||
| """passbook provider""" | """passbook provider""" | ||||||
| from allauth.socialaccount.providers.oauth2.urls import default_urlpatterns | from allauth.socialaccount.providers.oauth2.urls import default_urlpatterns | ||||||
|  | 
 | ||||||
| from allauth_passbook.provider import PassbookProvider | from allauth_passbook.provider import PassbookProvider | ||||||
| 
 | 
 | ||||||
| urlpatterns = default_urlpatterns(PassbookProvider) | urlpatterns = default_urlpatterns(PassbookProvider) | ||||||
| @ -1,10 +1,10 @@ | |||||||
| """passbook adapter""" | """passbook adapter""" | ||||||
| import requests | import requests | ||||||
| 
 |  | ||||||
| from allauth.socialaccount import app_settings | from allauth.socialaccount import app_settings | ||||||
| from allauth.socialaccount.providers.oauth2.views import (OAuth2Adapter, | from allauth.socialaccount.providers.oauth2.views import (OAuth2Adapter, | ||||||
|                                                           OAuth2CallbackView, |                                                           OAuth2CallbackView, | ||||||
|                                                           OAuth2LoginView) |                                                           OAuth2LoginView) | ||||||
|  | 
 | ||||||
| from allauth_passbook.provider import PassbookProvider | from allauth_passbook.provider import PassbookProvider | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -3,7 +3,7 @@ from setuptools import setup | |||||||
| 
 | 
 | ||||||
| setup( | setup( | ||||||
|     name='django-allauth-passbook', |     name='django-allauth-passbook', | ||||||
|     version='1.0.0', |     version='0.2.0-beta', | ||||||
|     description='passbook support for django-allauth', |     description='passbook support for django-allauth', | ||||||
|     # long_description='\n'.join(read_simple('docs/index.md')[2:]), |     # long_description='\n'.join(read_simple('docs/index.md')[2:]), | ||||||
|     long_description_content_type='text/markdown', |     long_description_content_type='text/markdown', | ||||||
							
								
								
									
										5
									
								
								client-packages/sentry-auth-passbook/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								client-packages/sentry-auth-passbook/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | *.pyc | ||||||
|  | *.egg-info/ | ||||||
|  | *.eggs | ||||||
|  | /dist | ||||||
|  | /build | ||||||
							
								
								
									
										32
									
								
								client-packages/sentry-auth-passbook/.travis.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								client-packages/sentry-auth-passbook/.travis.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | sudo: false | ||||||
|  | language: python | ||||||
|  | services: | ||||||
|  |   - memcached | ||||||
|  |   - postgresql | ||||||
|  |   - redis-server | ||||||
|  | python: | ||||||
|  |   - '2.7' | ||||||
|  | cache: | ||||||
|  |   directories: | ||||||
|  |     - node_modules | ||||||
|  |     - "$HOME/.cache/pip" | ||||||
|  | deploy: | ||||||
|  |   provider: pypi | ||||||
|  |   user: getsentry | ||||||
|  |   password: | ||||||
|  |     secure: kVmxKHkBWRLYyZme05p+WZSJmb8GjHV9uyuaSCVMRlqWCW+GXRB7P1xXR2jb9URTlNdcs56Ab/UrwzCbMFGC8LmwCeFVgIR/ltytVZG2FgXZPWaeA4dH25qK2oGWgzJ/xeiMpmuJqN9hRl25MX6jG7FZKvrrOkG7+8tpPd1yO+uYWZQbnebZMjcPBqEpn7CC0hR39GSoyVAbydpMe5hwENGQM26CepcicdrelfawItoUrXrkJzBHkIQQTO/xRSbCtRJOtzI5lwtv3GP0hcbOy5tI5dhG/93pLwZRc5+dZaCaP7oaVeOcBjN0zfINRQobt8d6h2Qgvd/YyFkGi0/xKn1zMmKIVLOG6VsYwEAUq8wNOsP4A/jdm4Y0J/1oEZStCkpaGpx85TYi4kq1hWQdyqaVJSPhh4Tk4roIaS2zOYQl+nIpbHqmJ4FJrg1il+TCdjBXobATQ1mKRBUrjD+RDzH/r4ogbd8+UwvvvevpqS2K+/wgT6UD0MzDInv9S29CUQvuFhPoqyJb5XRddHMRE9EEK/2Z8tFN91sDATnqfXHgwnvu00q/nKP5JnijBPzGmx7ydgUViIukklDrlPvo9BbRJz0Vr2vbAvMTrLMLCXqi5CwTm+v+iaOf/YaCziaG2vx0eVASYjpOLCedSgRZBubPM8z4E/HMXhChN7sVDWk= | ||||||
|  |   on: | ||||||
|  |     tags: true | ||||||
|  |   distributions: sdist bdist_wheel | ||||||
|  | env: | ||||||
|  |   global: | ||||||
|  |     - PIP_DOWNLOAD_CACHE=".pip_download_cache" | ||||||
|  | before_install: | ||||||
|  |   - pip install codecov | ||||||
|  | install: | ||||||
|  |   - make develop | ||||||
|  | script: | ||||||
|  |   - PYFLAKES_NODOCTEST=1 flake8 | ||||||
|  |   - coverage run --source=. -m py.test tests | ||||||
|  | after_success: | ||||||
|  |   - codecov | ||||||
							
								
								
									
										201
									
								
								client-packages/sentry-auth-passbook/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								client-packages/sentry-auth-passbook/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,201 @@ | |||||||
|  |                               Apache License | ||||||
|  |                         Version 2.0, January 2004 | ||||||
|  |                      http://www.apache.org/licenses/ | ||||||
|  |  | ||||||
|  | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
|  |  | ||||||
|  | 1. Definitions. | ||||||
|  |  | ||||||
|  |    "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |    and distribution as defined by Sections 1 through 9 of this document. | ||||||
|  |  | ||||||
|  |    "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|  |    the copyright owner that is granting the License. | ||||||
|  |  | ||||||
|  |    "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |    other entities that control, are controlled by, or are under common | ||||||
|  |    control with that entity. For the purposes of this definition, | ||||||
|  |    "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |    direction or management of such entity, whether by contract or | ||||||
|  |    otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |    outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
|  |  | ||||||
|  |    "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |    exercising permissions granted by this License. | ||||||
|  |  | ||||||
|  |    "Source" form shall mean the preferred form for making modifications, | ||||||
|  |    including but not limited to software source code, documentation | ||||||
|  |    source, and configuration files. | ||||||
|  |  | ||||||
|  |    "Object" form shall mean any form resulting from mechanical | ||||||
|  |    transformation or translation of a Source form, including but | ||||||
|  |    not limited to compiled object code, generated documentation, | ||||||
|  |    and conversions to other media types. | ||||||
|  |  | ||||||
|  |    "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |    Object form, made available under the License, as indicated by a | ||||||
|  |    copyright notice that is included in or attached to the work | ||||||
|  |    (an example is provided in the Appendix below). | ||||||
|  |  | ||||||
|  |    "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |    form, that is based on (or derived from) the Work and for which the | ||||||
|  |    editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |    represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |    of this License, Derivative Works shall not include works that remain | ||||||
|  |    separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |    the Work and Derivative Works thereof. | ||||||
|  |  | ||||||
|  |    "Contribution" shall mean any work of authorship, including | ||||||
|  |    the original version of the Work and any modifications or additions | ||||||
|  |    to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |    submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |    or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |    the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |    means any form of electronic, verbal, or written communication sent | ||||||
|  |    to the Licensor or its representatives, including but not limited to | ||||||
|  |    communication on electronic mailing lists, source code control systems, | ||||||
|  |    and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |    Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |    excluding communication that is conspicuously marked or otherwise | ||||||
|  |    designated in writing by the copyright owner as "Not a Contribution." | ||||||
|  |  | ||||||
|  |    "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |    on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |    subsequently incorporated within the Work. | ||||||
|  |  | ||||||
|  | 2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
|  |    this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |    worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |    copyright license to reproduce, prepare Derivative Works of, | ||||||
|  |    publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |    Work and such Derivative Works in Source or Object form. | ||||||
|  |  | ||||||
|  | 3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |    this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |    worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |    (except as stated in this section) patent license to make, have made, | ||||||
|  |    use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |    where such license applies only to those patent claims licensable | ||||||
|  |    by such Contributor that are necessarily infringed by their | ||||||
|  |    Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |    with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |    institute patent litigation against any entity (including a | ||||||
|  |    cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |    or a Contribution incorporated within the Work constitutes direct | ||||||
|  |    or contributory patent infringement, then any patent licenses | ||||||
|  |    granted to You under this License for that Work shall terminate | ||||||
|  |    as of the date such litigation is filed. | ||||||
|  |  | ||||||
|  | 4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |    Work or Derivative Works thereof in any medium, with or without | ||||||
|  |    modifications, and in Source or Object form, provided that You | ||||||
|  |    meet the following conditions: | ||||||
|  |  | ||||||
|  |    (a) You must give any other recipients of the Work or | ||||||
|  |        Derivative Works a copy of this License; and | ||||||
|  |  | ||||||
|  |    (b) You must cause any modified files to carry prominent notices | ||||||
|  |        stating that You changed the files; and | ||||||
|  |  | ||||||
|  |    (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |        that You distribute, all copyright, patent, trademark, and | ||||||
|  |        attribution notices from the Source form of the Work, | ||||||
|  |        excluding those notices that do not pertain to any part of | ||||||
|  |        the Derivative Works; and | ||||||
|  |  | ||||||
|  |    (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |        distribution, then any Derivative Works that You distribute must | ||||||
|  |        include a readable copy of the attribution notices contained | ||||||
|  |        within such NOTICE file, excluding those notices that do not | ||||||
|  |        pertain to any part of the Derivative Works, in at least one | ||||||
|  |        of the following places: within a NOTICE text file distributed | ||||||
|  |        as part of the Derivative Works; within the Source form or | ||||||
|  |        documentation, if provided along with the Derivative Works; or, | ||||||
|  |        within a display generated by the Derivative Works, if and | ||||||
|  |        wherever such third-party notices normally appear. The contents | ||||||
|  |        of the NOTICE file are for informational purposes only and | ||||||
|  |        do not modify the License. You may add Your own attribution | ||||||
|  |        notices within Derivative Works that You distribute, alongside | ||||||
|  |        or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |        that such additional attribution notices cannot be construed | ||||||
|  |        as modifying the License. | ||||||
|  |  | ||||||
|  |    You may add Your own copyright statement to Your modifications and | ||||||
|  |    may provide additional or different license terms and conditions | ||||||
|  |    for use, reproduction, or distribution of Your modifications, or | ||||||
|  |    for any such Derivative Works as a whole, provided Your use, | ||||||
|  |    reproduction, and distribution of the Work otherwise complies with | ||||||
|  |    the conditions stated in this License. | ||||||
|  |  | ||||||
|  | 5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |    any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |    by You to the Licensor shall be under the terms and conditions of | ||||||
|  |    this License, without any additional terms or conditions. | ||||||
|  |    Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |    the terms of any separate license agreement you may have executed | ||||||
|  |    with Licensor regarding such Contributions. | ||||||
|  |  | ||||||
|  | 6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |    names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |    except as required for reasonable and customary use in describing the | ||||||
|  |    origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  |  | ||||||
|  | 7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |    agreed to in writing, Licensor provides the Work (and each | ||||||
|  |    Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |    implied, including, without limitation, any warranties or conditions | ||||||
|  |    of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |    PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |    appropriateness of using or redistributing the Work and assume any | ||||||
|  |    risks associated with Your exercise of permissions under this License. | ||||||
|  |  | ||||||
|  | 8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |    whether in tort (including negligence), contract, or otherwise, | ||||||
|  |    unless required by applicable law (such as deliberate and grossly | ||||||
|  |    negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |    liable to You for damages, including any direct, indirect, special, | ||||||
|  |    incidental, or consequential damages of any character arising as a | ||||||
|  |    result of this License or out of the use or inability to use the | ||||||
|  |    Work (including but not limited to damages for loss of goodwill, | ||||||
|  |    work stoppage, computer failure or malfunction, or any and all | ||||||
|  |    other commercial damages or losses), even if such Contributor | ||||||
|  |    has been advised of the possibility of such damages. | ||||||
|  |  | ||||||
|  | 9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |    the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |    and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |    or other liability obligations and/or rights consistent with this | ||||||
|  |    License. However, in accepting such obligations, You may act only | ||||||
|  |    on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |    of any other Contributor, and only if You agree to indemnify, | ||||||
|  |    defend, and hold each Contributor harmless for any liability | ||||||
|  |    incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |    of your accepting any such warranty or additional liability. | ||||||
|  |  | ||||||
|  | END OF TERMS AND CONDITIONS | ||||||
|  |  | ||||||
|  | APPENDIX: How to apply the Apache License to your work. | ||||||
|  |  | ||||||
|  |    To apply the Apache License to your work, attach the following | ||||||
|  |    boilerplate notice, with the fields enclosed by brackets "[]" | ||||||
|  |    replaced with your own identifying information. (Don't include | ||||||
|  |    the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |    comment syntax for the file format. We also recommend that a | ||||||
|  |    file or class name and description of purpose be included on the | ||||||
|  |    same "printed page" as the copyright notice for easier | ||||||
|  |    identification within third-party archives. | ||||||
|  |  | ||||||
|  | Copyright 2016 Functional Software, Inc. | ||||||
|  |  | ||||||
|  | Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | you may not use this file except in compliance with the License. | ||||||
|  | You may obtain a copy of the License at | ||||||
|  |  | ||||||
|  |     http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  |  | ||||||
|  | Unless required by applicable law or agreed to in writing, software | ||||||
|  | distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | See the License for the specific language governing permissions and | ||||||
|  | limitations under the License. | ||||||
							
								
								
									
										3
									
								
								client-packages/sentry-auth-passbook/MANIFEST.in
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								client-packages/sentry-auth-passbook/MANIFEST.in
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | include setup.py package.json webpack.config.js README.rst MANIFEST.in LICENSE AUTHORS | ||||||
|  | recursive-include sentry_auth_supervisr/templates * | ||||||
|  | global-exclude *~ | ||||||
							
								
								
									
										26
									
								
								client-packages/sentry-auth-passbook/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								client-packages/sentry-auth-passbook/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | .PHONY: clean develop install-tests lint publish test | ||||||
|  |  | ||||||
|  | develop: | ||||||
|  | 	pip install "pip>=7" | ||||||
|  | 	pip install -e . | ||||||
|  | 	make install-tests | ||||||
|  |  | ||||||
|  | install-tests: | ||||||
|  | 	pip install .[tests] | ||||||
|  |  | ||||||
|  | lint: | ||||||
|  | 	@echo "--> Linting python" | ||||||
|  | 	flake8 | ||||||
|  | 	@echo "" | ||||||
|  |  | ||||||
|  | test: | ||||||
|  | 	@echo "--> Running Python tests" | ||||||
|  | 	py.test tests || exit 1 | ||||||
|  | 	@echo "" | ||||||
|  |  | ||||||
|  | publish: | ||||||
|  | 	python setup.py sdist bdist_wheel upload | ||||||
|  |  | ||||||
|  | clean: | ||||||
|  | 	rm -rf *.egg-info src/*.egg-info | ||||||
|  | 	rm -rf dist build | ||||||
							
								
								
									
										55
									
								
								client-packages/sentry-auth-passbook/README.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								client-packages/sentry-auth-passbook/README.rst
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | |||||||
|  | GitHub Auth for Sentry | ||||||
|  | ====================== | ||||||
|  |  | ||||||
|  | An SSO provider for Sentry which enables GitHub organization-restricted authentication. | ||||||
|  |  | ||||||
|  | Install | ||||||
|  | ------- | ||||||
|  |  | ||||||
|  | :: | ||||||
|  |  | ||||||
|  |     $ pip install https://github.com/getsentry/sentry-auth-github/archive/master.zip | ||||||
|  |  | ||||||
|  | Setup | ||||||
|  | ----- | ||||||
|  |  | ||||||
|  | Create a new application under your organization in GitHub. Enter the **Authorization | ||||||
|  | callback URL** as the prefix to your Sentry installation: | ||||||
|  |  | ||||||
|  | :: | ||||||
|  |  | ||||||
|  |     https://example.sentry.com | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Once done, grab your API keys and drop them in your ``sentry.conf.py``: | ||||||
|  |  | ||||||
|  | .. code-block:: python | ||||||
|  |  | ||||||
|  |     GITHUB_APP_ID = "" | ||||||
|  |  | ||||||
|  |     GITHUB_API_SECRET = "" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Verified email addresses can optionally be required: | ||||||
|  |  | ||||||
|  | .. code-block:: python | ||||||
|  |  | ||||||
|  |     GITHUB_REQUIRE_VERIFIED_EMAIL = True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Optionally you may also specify the domain (for GHE users): | ||||||
|  |  | ||||||
|  | .. code-block:: python | ||||||
|  |  | ||||||
|  |     GITHUB_BASE_DOMAIN = "git.example.com" | ||||||
|  |  | ||||||
|  |     GITHUB_API_DOMAIN = "api.git.example.com" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | If Subdomain isolation is disabled in GHE: | ||||||
|  |  | ||||||
|  | .. code-block:: python | ||||||
|  |  | ||||||
|  |     GITHUB_BASE_DOMAIN = "git.example.com" | ||||||
|  |  | ||||||
|  |     GITHUB_API_DOMAIN = "git.example.com/api/v3" | ||||||
							
								
								
									
										14
									
								
								client-packages/sentry-auth-passbook/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								client-packages/sentry-auth-passbook/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | from __future__ import absolute_import | ||||||
|  |  | ||||||
|  | # Run tests against sqlite for simplicity | ||||||
|  | import os | ||||||
|  | import os.path | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | sys.path.insert(0, os.path.join(os.path.dirname(__file__))) | ||||||
|  |  | ||||||
|  | os.environ.setdefault('DB', 'sqlite') | ||||||
|  |  | ||||||
|  | pytest_plugins = [ | ||||||
|  |     'sentry.utils.pytest' | ||||||
|  | ] | ||||||
| @ -0,0 +1,7 @@ | |||||||
|  | from __future__ import absolute_import | ||||||
|  |  | ||||||
|  | from sentry.auth import register | ||||||
|  |  | ||||||
|  | from .provider import PassbookOAuth2Provider | ||||||
|  |  | ||||||
|  | register('passbook', PassbookOAuth2Provider) | ||||||
| @ -0,0 +1,45 @@ | |||||||
|  | from __future__ import absolute_import, print_function | ||||||
|  |  | ||||||
|  | from requests.exceptions import RequestException | ||||||
|  |  | ||||||
|  | from sentry import http | ||||||
|  | from sentry.utils import json | ||||||
|  |  | ||||||
|  | from .constants import BASE_DOMAIN | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PassbookApiError(Exception): | ||||||
|  |     def __init__(self, message='', status=0): | ||||||
|  |         super(PassbookApiError, self).__init__(message) | ||||||
|  |         self.status = status | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PassbookClient(object): | ||||||
|  |     def __init__(self, client_id, client_secret): | ||||||
|  |         self.client_id = client_id | ||||||
|  |         self.client_secret = client_secret | ||||||
|  |         self.http = http.build_session() | ||||||
|  |  | ||||||
|  |     def _request(self, path, access_token): | ||||||
|  |         params = { | ||||||
|  |             'client_id': self.client_id, | ||||||
|  |             'client_secret': self.client_secret, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         headers = { | ||||||
|  |             'Authorization': 'Bearer {0}'.format(access_token), | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             req = self.http.get('https://{0}/{1}'.format(BASE_DOMAIN, path.lstrip('/')), | ||||||
|  |                 params=params, | ||||||
|  |                 headers=headers, | ||||||
|  |             ) | ||||||
|  |         except RequestException as e: | ||||||
|  |             raise PassbookApiError(unicode(e), status=getattr(e, 'status_code', 0)) | ||||||
|  |         if req.status_code < 200 or req.status_code >= 300: | ||||||
|  |             raise PassbookApiError(req.content, status=req.status_code) | ||||||
|  |         return json.loads(req.content) | ||||||
|  |  | ||||||
|  |     def get_user(self, access_token): | ||||||
|  |         return self._request('/api/v1/openid/', access_token) | ||||||
| @ -0,0 +1,14 @@ | |||||||
|  | from __future__ import absolute_import, print_function | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  |  | ||||||
|  | CLIENT_ID = getattr(settings, 'PASSBOOK_APP_ID', None) | ||||||
|  |  | ||||||
|  | CLIENT_SECRET = getattr(settings, 'PASSBOOK_API_SECRET', None) | ||||||
|  |  | ||||||
|  | SCOPE = 'openid:userinfo' | ||||||
|  |  | ||||||
|  | BASE_DOMAIN = getattr(settings, 'PASSBOOK_BASE_DOMAIN', 'id.beryju.org') | ||||||
|  |  | ||||||
|  | ACCESS_TOKEN_URL = 'https://{0}/application/oauth/token/'.format(BASE_DOMAIN) | ||||||
|  | AUTHORIZE_URL = 'https://{0}/application/oauth/authorize/'.format(BASE_DOMAIN) | ||||||
| @ -0,0 +1,62 @@ | |||||||
|  | from __future__ import absolute_import, print_function | ||||||
|  |  | ||||||
|  | from sentry.auth.exceptions import IdentityNotValid | ||||||
|  | from sentry.auth.providers.oauth2 import (OAuth2Callback, OAuth2Login, | ||||||
|  |                                           OAuth2Provider) | ||||||
|  |  | ||||||
|  | from .client import PassbookApiError, PassbookClient | ||||||
|  | from .constants import (ACCESS_TOKEN_URL, AUTHORIZE_URL, CLIENT_ID, | ||||||
|  |                         CLIENT_SECRET, SCOPE) | ||||||
|  | from .views import FetchUser, PassbookConfigureView | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PassbookOAuth2Provider(OAuth2Provider): | ||||||
|  |     access_token_url = ACCESS_TOKEN_URL | ||||||
|  |     authorize_url = AUTHORIZE_URL | ||||||
|  |     name = 'Passbook' | ||||||
|  |     client_id = CLIENT_ID | ||||||
|  |     client_secret = CLIENT_SECRET | ||||||
|  |  | ||||||
|  |     def __init__(self, **config): | ||||||
|  |         super(PassbookOAuth2Provider, self).__init__(**config) | ||||||
|  |  | ||||||
|  |     def get_configure_view(self): | ||||||
|  |         return PassbookConfigureView.as_view() | ||||||
|  |  | ||||||
|  |     def get_auth_pipeline(self): | ||||||
|  |         return [ | ||||||
|  |             OAuth2Login( | ||||||
|  |                 authorize_url=self.authorize_url, | ||||||
|  |                 client_id=self.client_id, | ||||||
|  |                 scope=SCOPE, | ||||||
|  |             ), | ||||||
|  |             OAuth2Callback( | ||||||
|  |                 access_token_url=self.access_token_url, | ||||||
|  |                 client_id=self.client_id, | ||||||
|  |                 client_secret=self.client_secret, | ||||||
|  |             ), | ||||||
|  |             FetchUser( | ||||||
|  |                 client_id=self.client_id, | ||||||
|  |                 client_secret=self.client_secret, | ||||||
|  |             ), | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def get_refresh_token_url(self): | ||||||
|  |         return ACCESS_TOKEN_URL | ||||||
|  |  | ||||||
|  |     def build_identity(self, state): | ||||||
|  |         data = state['data'] | ||||||
|  |         user_data = state['user'] | ||||||
|  |         return { | ||||||
|  |             'id': user_data['email'], | ||||||
|  |             'email': user_data['email'], | ||||||
|  |             'name': user_data['name'], | ||||||
|  |             'data': self.get_oauth_data(data), | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def build_config(self, state): | ||||||
|  |         return {} | ||||||
|  |  | ||||||
|  |     def refresh_identity(self, auth_identity): | ||||||
|  |         client = PassbookClient(self.client_id, self.client_secret) | ||||||
|  |         access_token = auth_identity.data['access_token'] | ||||||
| @ -0,0 +1,75 @@ | |||||||
|  | from __future__ import absolute_import, print_function | ||||||
|  |  | ||||||
|  | from django import forms | ||||||
|  |  | ||||||
|  | from sentry.auth.view import AuthView, ConfigureView | ||||||
|  | from sentry.models import AuthIdentity | ||||||
|  |  | ||||||
|  | from .client import PassbookClient | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _get_name_from_email(email): | ||||||
|  |     """ | ||||||
|  |     Given an email return a capitalized name. Ex. john.smith@example.com would return John Smith. | ||||||
|  |     """ | ||||||
|  |     name = email.rsplit('@', 1)[0] | ||||||
|  |     name = ' '.join([n_part.capitalize() for n_part in name.split('.')]) | ||||||
|  |     return name | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FetchUser(AuthView): | ||||||
|  |     def __init__(self, client_id, client_secret, *args, **kwargs): | ||||||
|  |         self.client = PassbookClient(client_id, client_secret) | ||||||
|  |         super(FetchUser, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def handle(self, request, helper): | ||||||
|  |         access_token = helper.fetch_state('data')['access_token'] | ||||||
|  |  | ||||||
|  |         user = self.client.get_user(access_token) | ||||||
|  |  | ||||||
|  |         # A user hasn't set their name in their Passbook profile so it isn't | ||||||
|  |         # populated in the response | ||||||
|  |         if not user.get('name'): | ||||||
|  |             user['name'] = _get_name_from_email(user['email']) | ||||||
|  |  | ||||||
|  |         helper.bind_state('user', user) | ||||||
|  |  | ||||||
|  |         return helper.next_step() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConfirmEmailForm(forms.Form): | ||||||
|  |     email = forms.EmailField(label='Email') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConfirmEmail(AuthView): | ||||||
|  |     def handle(self, request, helper): | ||||||
|  |         user = helper.fetch_state('user') | ||||||
|  |  | ||||||
|  |         # TODO(dcramer): this isnt ideal, but our current flow doesnt really | ||||||
|  |         # support this behavior; | ||||||
|  |         try: | ||||||
|  |             auth_identity = AuthIdentity.objects.select_related('user').get( | ||||||
|  |                 auth_provider=helper.auth_provider, | ||||||
|  |                 ident=user['id'], | ||||||
|  |             ) | ||||||
|  |         except AuthIdentity.DoesNotExist: | ||||||
|  |             pass | ||||||
|  |         else: | ||||||
|  |             user['email'] = auth_identity.user.email | ||||||
|  |  | ||||||
|  |         if user.get('email'): | ||||||
|  |             return helper.next_step() | ||||||
|  |  | ||||||
|  |         form = ConfirmEmailForm(request.POST or None) | ||||||
|  |         if form.is_valid(): | ||||||
|  |             user['email'] = form.cleaned_data['email'] | ||||||
|  |             helper.bind_state('user', user) | ||||||
|  |             return helper.next_step() | ||||||
|  |  | ||||||
|  |         return self.respond('sentry_auth_passbook/enter-email.html', { | ||||||
|  |             'form': form, | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  | class PassbookConfigureView(ConfigureView): | ||||||
|  |     def dispatch(self, request, organization, auth_provider): | ||||||
|  |         return self.render('sentry_auth_passbook/configure.html') | ||||||
							
								
								
									
										12
									
								
								client-packages/sentry-auth-passbook/setup.cfg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								client-packages/sentry-auth-passbook/setup.cfg
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | [wheel] | ||||||
|  | universal = 1 | ||||||
|  |  | ||||||
|  | [pytest] | ||||||
|  | python_files = test*.py | ||||||
|  | addopts = --tb=native -p no:doctest | ||||||
|  | norecursedirs = bin dist docs htmlcov script hooks node_modules .* {args} | ||||||
|  |  | ||||||
|  | [flake8] | ||||||
|  | ignore = F999,E501,E128,E124,E402,W503,E731,C901 | ||||||
|  | max-line-length = 100 | ||||||
|  | exclude = .tox,.git,*/migrations/*,node_modules/*,docs/* | ||||||
							
								
								
									
										45
									
								
								client-packages/sentry-auth-passbook/setup.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								client-packages/sentry-auth-passbook/setup.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,45 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  | """ | ||||||
|  | sentry-auth-passbook | ||||||
|  | ================== | ||||||
|  |  | ||||||
|  | :copyright: (c) 2016 Functional Software, Inc | ||||||
|  | """ | ||||||
|  | from setuptools import find_packages, setup | ||||||
|  |  | ||||||
|  | install_requires = [ | ||||||
|  |     'sentry>=7.0.0', | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | tests_require = [ | ||||||
|  |     'mock', | ||||||
|  |     'flake8>=2.0,<2.1', | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | setup( | ||||||
|  |     name='sentry-auth-passbook', | ||||||
|  |     version='0.2.0-beta', | ||||||
|  |     author='BeryJu.org', | ||||||
|  |     author_email='support@beryju.org', | ||||||
|  |     url='https://passbook.beryju.org', | ||||||
|  |     description='passbook authentication provider for Sentry', | ||||||
|  |     long_description=__doc__, | ||||||
|  |     license='MIT', | ||||||
|  |     packages=find_packages(exclude=['tests']), | ||||||
|  |     zip_safe=False, | ||||||
|  |     install_requires=install_requires, | ||||||
|  |     tests_require=tests_require, | ||||||
|  |     extras_require={'tests': tests_require}, | ||||||
|  |     include_package_data=True, | ||||||
|  |     entry_points={ | ||||||
|  |         'sentry.apps': [ | ||||||
|  |             'auth_passbook = sentry_auth_passbook', | ||||||
|  |         ], | ||||||
|  |     }, | ||||||
|  |     classifiers=[ | ||||||
|  |         'Intended Audience :: Developers', | ||||||
|  |         'Intended Audience :: System Administrators', | ||||||
|  |         'Operating System :: OS Independent', | ||||||
|  |         'Topic :: Software Development' | ||||||
|  |     ], | ||||||
|  | ) | ||||||
| @ -0,0 +1,6 @@ | |||||||
|  | from sentry.testutils import TestCase | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GitHubOAuth2ProviderTest(TestCase): | ||||||
|  |     def test_simple(self): | ||||||
|  |         pass | ||||||
							
								
								
									
										17
									
								
								client-packages/sentry-auth-passbook/tests/test_views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								client-packages/sentry-auth-passbook/tests/test_views.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | from __future__ import absolute_import, print_function | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from sentry_auth_sentry.views import _get_name_from_email | ||||||
|  |  | ||||||
|  | expected_data = [ | ||||||
|  |     ('john.smith@example.com', 'John Smith'), | ||||||
|  |     ('john@example.com', 'John'), | ||||||
|  |     ('XYZ-234=3523@example.com', 'Xyz-234=3523'), | ||||||
|  |     ('XYZ.1111@example.com', 'Xyz 1111'), | ||||||
|  |     ('JOHN@example.com', 'John'), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.parametrize("email,expected_name", expected_data) | ||||||
|  | def test_get_name_from_email(email, expected_name): | ||||||
|  |     assert _get_name_from_email(email) == expected_name | ||||||
							
								
								
									
										25
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							| @ -1,25 +0,0 @@ | |||||||
| passbook (0.1.7) stable; urgency=medium |  | ||||||
|  |  | ||||||
|   * bump version: 0.1.3-beta -> 0.1.4-beta |  | ||||||
|   * implicitly add kubernetes-healthcheck-host in helm configmap |  | ||||||
|   * fix debian build (again) |  | ||||||
|   * add PropertyMapping Model, add Subclass for SAML, test with AWS |  | ||||||
|   * add custom DynamicArrayField to better handle arrays |  | ||||||
|   * format data before inserting it |  | ||||||
|   * bump version: 0.1.4-beta -> 0.1.5-beta |  | ||||||
|   * fix static files missing for debian package |  | ||||||
|   * fix password not getting set on user import |  | ||||||
|   * remove audit's login attempt |  | ||||||
|   * add passing property to PolicyEngine |  | ||||||
|   * fix captcha factor not loading keys from Factor class |  | ||||||
|   * bump version: 0.1.5-beta -> 0.1.6-beta |  | ||||||
|   * fix MATCH_EXACT not working as intended |  | ||||||
|   * Improve access control for saml |  | ||||||
|  |  | ||||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Fri, 08 Mar 2019 20:37:05 +0000 |  | ||||||
|  |  | ||||||
| passbook (0.1.4) stable; urgency=medium |  | ||||||
|  |  | ||||||
|   * initial debian package release |  | ||||||
|  |  | ||||||
|  -- Jens Langhammer <jens.langhammer@beryju.org>  Wed, 06 Mar 2019 18:22:41 +0000 |  | ||||||
							
								
								
									
										1
									
								
								debian/compat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/compat
									
									
									
									
										vendored
									
									
								
							| @ -1 +0,0 @@ | |||||||
| 10 |  | ||||||
							
								
								
									
										20
									
								
								debian/config
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								debian/config
									
									
									
									
										vendored
									
									
								
							| @ -1,20 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
| # config maintainer script for passbook |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| # source debconf stuff |  | ||||||
| . /usr/share/debconf/confmodule |  | ||||||
|  |  | ||||||
| dbc_first_version=1.0.0 |  | ||||||
| dbc_dbuser=passbook |  | ||||||
| dbc_dbname=passbook |  | ||||||
|  |  | ||||||
| # source dbconfig-common shell library, and call the hook function |  | ||||||
| if [ -f /usr/share/dbconfig-common/dpkg/config.pgsql ]; then |  | ||||||
|     . /usr/share/dbconfig-common/dpkg/config.pgsql |  | ||||||
|     dbc_go passbook "$@" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
|  |  | ||||||
| exit 0 |  | ||||||
							
								
								
									
										14
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/control
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | |||||||
| Source: passbook |  | ||||||
| Section: admin |  | ||||||
| Priority: optional |  | ||||||
| Maintainer: BeryJu.org <support@beryju.org> |  | ||||||
| Uploaders: Jens Langhammer <jens@beryju.org>, BeryJu.org <support@beryju.org> |  | ||||||
| Build-Depends: debhelper (>= 10), dh-systemd (>= 1.5), dh-exec, wget, dh-exec, python3 (>= 3.5) | python3.6 | python3.7 |  | ||||||
| Standards-Version: 3.9.6 |  | ||||||
|  |  | ||||||
| Package: passbook |  | ||||||
| Architecture: all |  | ||||||
| Recommends: mysql-server, redis-server |  | ||||||
| Pre-Depends: adduser, libldap2-dev, libsasl2-dev |  | ||||||
| Depends: python3 (>= 3.5) | python3.6 | python3.7, python3-pip, dbconfig-pgsql | dbconfig-no-thanks, ${misc:Depends} |  | ||||||
| Description: Authentication Provider/Proxy supporting protocols like SAML, OAuth, LDAP and more. |  | ||||||
							
								
								
									
										22
									
								
								debian/copyright
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								debian/copyright
									
									
									
									
										vendored
									
									
								
							| @ -1,22 +0,0 @@ | |||||||
| MIT License |  | ||||||
|  |  | ||||||
| Copyright (c) 2019 BeryJu.org |  | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
| of this software and associated documentation files (the "Software"), to deal |  | ||||||
| in the Software without restriction, including without limitation the rights |  | ||||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
| copies of the Software, and to permit persons to whom the Software is |  | ||||||
| furnished to do so, subject to the following conditions: |  | ||||||
|  |  | ||||||
| The above copyright notice and this permission notice shall be included in all |  | ||||||
| copies or substantial portions of the Software. |  | ||||||
|  |  | ||||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
| SOFTWARE. |  | ||||||
|  |  | ||||||
							
								
								
									
										4
									
								
								debian/dirs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								debian/dirs
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +0,0 @@ | |||||||
| etc/passbook/ |  | ||||||
| etc/passbook/config.d/ |  | ||||||
| var/log/passbook/ |  | ||||||
| usr/share/passbook/ |  | ||||||
							
								
								
									
										44
									
								
								debian/etc/passbook/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										44
									
								
								debian/etc/passbook/config.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,44 +0,0 @@ | |||||||
| debug: false |  | ||||||
| http: |  | ||||||
|     host: 0.0.0.0 |  | ||||||
|     port: 8000 |  | ||||||
| secret_key_file: /etc/passbook/secret_key |  | ||||||
| log: |  | ||||||
|     level: |  | ||||||
|         console: INFO |  | ||||||
|         file: DEBUG |  | ||||||
|     file: /var/log/passbook/passbook.log |  | ||||||
| # Error reporting, disabled by default |  | ||||||
| # error_report_enabled: true |  | ||||||
|  |  | ||||||
| # Set this to the server's external address. |  | ||||||
| # This is used to generate external URLs |  | ||||||
| external_url: http://image.example.com |  | ||||||
|  |  | ||||||
| # This dictates how the Path is generated |  | ||||||
| # can be either of: |  | ||||||
| # - view_sha512_short |  | ||||||
| # - view_md5 |  | ||||||
| # - view_sha256 |  | ||||||
| # - view_sha512 |  | ||||||
| default_return_view: view_sha256 |  | ||||||
|  |  | ||||||
| # Set this to true if you only want to use external authentication |  | ||||||
| external_auth_only: false |  | ||||||
|  |  | ||||||
| # If this is true, images are automatically claimed if the windows user exists |  | ||||||
| # in django |  | ||||||
| auto_claim_enabled: true |  | ||||||
|  |  | ||||||
| # LDAP Authentication |  | ||||||
| # ldap: |  | ||||||
| #     enabled: false |  | ||||||
| #     server: |  | ||||||
| #         uri: 'ldap://dc1.example.com' |  | ||||||
| #         tls: false |  | ||||||
| #     bind: |  | ||||||
| #         dn: '' |  | ||||||
| #         password: '' |  | ||||||
| #     search_base: '' |  | ||||||
| #     filter: '(sAMAccountName=%(user)s)' |  | ||||||
| #     require_group: '' |  | ||||||
							
								
								
									
										2
									
								
								debian/gbp.conf
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								debian/gbp.conf
									
									
									
									
										vendored
									
									
								
							| @ -1,2 +0,0 @@ | |||||||
| [buildpackage] |  | ||||||
| export-dir=../build-area |  | ||||||
							
								
								
									
										8
									
								
								debian/install
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								debian/install
									
									
									
									
										vendored
									
									
								
							| @ -1,8 +0,0 @@ | |||||||
| passbook		/usr/share/passbook/ |  | ||||||
| static			/usr/share/passbook/ |  | ||||||
| manage.py		/usr/share/passbook/ |  | ||||||
| passbook.sh		/usr/share/passbook/ |  | ||||||
| vendor			/usr/share/passbook/ |  | ||||||
|  |  | ||||||
| debian/etc/passbook								/etc/ |  | ||||||
| debian/templates/database.yml					/usr/share/passbook/ |  | ||||||
							
								
								
									
										14
									
								
								debian/passbook-worker.service
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/passbook-worker.service
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | |||||||
| [Unit] |  | ||||||
| Description=passbook - Authentication Provider/Proxy (Background worker) |  | ||||||
| After=network.target |  | ||||||
| Requires=network.target |  | ||||||
|  |  | ||||||
| [Service] |  | ||||||
| User=passbook |  | ||||||
| Group=passbook |  | ||||||
| WorkingDirectory=/usr/share/passbook |  | ||||||
| Type=simple |  | ||||||
| ExecStart=/usr/share/passbook/passbook.sh worker |  | ||||||
|  |  | ||||||
| [Install] |  | ||||||
| WantedBy=multi-user.target |  | ||||||
							
								
								
									
										14
									
								
								debian/passbook.service
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								debian/passbook.service
									
									
									
									
										vendored
									
									
								
							| @ -1,14 +0,0 @@ | |||||||
| [Unit] |  | ||||||
| Description=passbook - Authentication Provider/Proxy |  | ||||||
| After=network.target |  | ||||||
| Requires=network.target |  | ||||||
|  |  | ||||||
| [Service] |  | ||||||
| User=passbook |  | ||||||
| Group=passbook |  | ||||||
| WorkingDirectory=/usr/share/passbook |  | ||||||
| Type=simple |  | ||||||
| ExecStart=/usr/share/passbook/passbook.sh web |  | ||||||
|  |  | ||||||
| [Install] |  | ||||||
| WantedBy=multi-user.target |  | ||||||
							
								
								
									
										36
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							| @ -1,36 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| . /usr/share/debconf/confmodule |  | ||||||
| . /usr/share/dbconfig-common/dpkg/postinst.pgsql |  | ||||||
|  |  | ||||||
| # you can set the default database encoding to something else |  | ||||||
| dbc_pgsql_createdb_encoding="UTF8" |  | ||||||
| dbc_generate_include=template:/etc/passbook/config.d/database.yml |  | ||||||
| dbc_generate_include_args="-o template_infile=/usr/share/passbook/database.yml" |  | ||||||
| dbc_go passbook "$@" |  | ||||||
|  |  | ||||||
| if [ -z "`getent group passbook`" ]; then |  | ||||||
| 	addgroup --quiet --system passbook |  | ||||||
| fi |  | ||||||
| if [ -z "`getent passwd passbook`" ]; then |  | ||||||
| 	echo " * Creating user and group passbook..." |  | ||||||
| 	adduser --quiet --system --home /usr/share/passbook --shell /bin/false --ingroup passbook --disabled-password --disabled-login --gecos "passbook User" passbook  >> /var/log/passbook/passbook.log 2>&1 |  | ||||||
| fi |  | ||||||
| echo " * Updating binary packages (psycopg2)" |  | ||||||
| python3 -m pip install --target=/usr/share/passbook/vendor/ --no-cache-dir --upgrade --force-reinstall psycopg2 >> /var/log/passbook/passbook.log 2>&1 |  | ||||||
| if [ ! -f '/etc/passbook/secret_key' ]; then |  | ||||||
| 	echo " * Generating Secret Key" |  | ||||||
| 	python3 -c 'import random; result = "".join([random.choice("abcdefghijklmnopqrstuvwxyz0123456789!@#$%^&*(-_=+)") for i in range(50)]); print(result)' > /etc/passbook/secret_key 2> /dev/null |  | ||||||
| fi |  | ||||||
| chown -R passbook: /usr/share/passbook/ |  | ||||||
| chown -R passbook: /etc/passbook/ |  | ||||||
| chown -R passbook: /var/log/passbook/ |  | ||||||
| chmod 440 /etc/passbook/secret_key |  | ||||||
| echo " * Running Database Migration" |  | ||||||
| /usr/share/passbook/passbook.sh migrate |  | ||||||
| echo " * A superuser can be created with this command '/usr/share/passbook/passbook.sh createsuperuser'" |  | ||||||
| echo " * You should probably also adjust your settings in '/etc/passbook/config.yml'" |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
							
								
								
									
										24
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							| @ -1,24 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| if [ -f /usr/share/debconf/confmodule ]; then |  | ||||||
|     . /usr/share/debconf/confmodule |  | ||||||
| fi |  | ||||||
| if [ -f /usr/share/dbconfig-common/dpkg/postrm.pgsql ]; then |  | ||||||
|     . /usr/share/dbconfig-common/dpkg/postrm.pgsql |  | ||||||
|     dbc_go passbook "$@" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if [ "$1" = "purge" ]; then |  | ||||||
|     if which ucf >/dev/null 2>&1; then |  | ||||||
|         ucf --purge /etc/passbook/config.d/database.yml |  | ||||||
|         ucfr --purge passbook /etc/passbook/config.d/database.yml |  | ||||||
|     fi |  | ||||||
|     rm -rf /etc/passbook/ |  | ||||||
|     rm -rf /usr/share/passbook/ |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								debian/prerm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								debian/prerm
									
									
									
									
										vendored
									
									
								
							| @ -1,10 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| . /usr/share/debconf/confmodule |  | ||||||
| . /usr/share/dbconfig-common/dpkg/prerm.pgsql |  | ||||||
| dbc_go passbook "$@" |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
|  |  | ||||||
							
								
								
									
										27
									
								
								debian/rules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								debian/rules
									
									
									
									
										vendored
									
									
								
							| @ -1,27 +0,0 @@ | |||||||
| #!/usr/bin/make -f |  | ||||||
|  |  | ||||||
| # Uncomment this to turn on verbose mode. |  | ||||||
| # export DH_VERBOSE=1 |  | ||||||
|  |  | ||||||
| %: |  | ||||||
| 	dh $@ --with=systemd |  | ||||||
|  |  | ||||||
| build-arch: |  | ||||||
| 	python3 -m pip install setuptools |  | ||||||
| 	python3 -m pip install --target=vendor/ -r requirements.txt |  | ||||||
|  |  | ||||||
| override_dh_strip: |  | ||||||
| 	dh_strip --exclude=psycopg2 |  | ||||||
|  |  | ||||||
| override_dh_shlibdeps: |  | ||||||
| 	dh_shlibdeps --exclude=psycopg2 |  | ||||||
|  |  | ||||||
| override_dh_installinit: |  | ||||||
| 	dh_installinit --name=passbook |  | ||||||
| 	dh_installinit --name=passbook-worker |  | ||||||
| 	dh_systemd_enable --name=passbook |  | ||||||
| 	dh_systemd_enable --name=passbook-worker |  | ||||||
| 	dh_systemd_start |  | ||||||
|  |  | ||||||
| # override_dh_usrlocal to do nothing |  | ||||||
| override_dh_usrlocal: |  | ||||||
							
								
								
									
										1
									
								
								debian/source/format
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/source/format
									
									
									
									
										vendored
									
									
								
							| @ -1 +0,0 @@ | |||||||
| 3.0 (native) |  | ||||||
							
								
								
									
										8
									
								
								debian/templates/database.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								debian/templates/database.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,8 +0,0 @@ | |||||||
| databases: |  | ||||||
|   default: |  | ||||||
|     engine: django.db.backends.postgresql |  | ||||||
|     name: _DBC_DBNAME_ |  | ||||||
|     user: _DBC_DBUSER_ |  | ||||||
|     password: _DBC_DBPASS_ |  | ||||||
|     host: _DBC_DBSERVER_ |  | ||||||
|     port: _DBC_DBPORT_ |  | ||||||
| @ -1,6 +1,6 @@ | |||||||
| apiVersion: v1 | apiVersion: v1 | ||||||
| appVersion: "0.1.8-beta" | appVersion: "0.2.0-beta" | ||||||
| description: A Helm chart for passbook. | description: A Helm chart for passbook. | ||||||
| name: passbook | name: passbook | ||||||
| version: "0.1.8-beta" | version: "0.2.0-beta" | ||||||
| icon: https://passbook.beryju.org/images/logo.png | icon: https://git.beryju.org/uploads/-/system/project/avatar/108/logo.png | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								helm/passbook/app-readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								helm/passbook/app-readme.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | # passbook | ||||||
							
								
								
									
										
											BIN
										
									
								
								helm/passbook/charts/rabbitmq-4.3.2.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								helm/passbook/charts/rabbitmq-4.3.2.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										98
									
								
								helm/passbook/questions.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								helm/passbook/questions.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,98 @@ | |||||||
|  | --- | ||||||
|  | categories: | ||||||
|  |   - Authentication | ||||||
|  |   - SSO | ||||||
|  | questions: | ||||||
|  |   - default: "true" | ||||||
|  |     variable: config.error_reporting | ||||||
|  |     type: boolean | ||||||
|  |     description: "Enable error-reporting to sentry.services.beryju.org" | ||||||
|  |     group: "passbook Configuration" | ||||||
|  |     label: "Error Reporting" | ||||||
|  |     #################################################################### | ||||||
|  |     ### PostgreSQL | ||||||
|  |     #################################################################### | ||||||
|  |   - variable: postgresql.enabled | ||||||
|  |     default: true | ||||||
|  |     description: "Deploy a database server as part of this deployment, or set to false and configure an external database connection." | ||||||
|  |     type: boolean | ||||||
|  |     required: true | ||||||
|  |     label: Install PostgreSQL | ||||||
|  |     show_subquestion_if: true | ||||||
|  |     group: "Database Settings" | ||||||
|  |     subquestions: | ||||||
|  |       - variable: postgresql.postgresqlDatabase | ||||||
|  |         default: "passbook" | ||||||
|  |         description: "Database name to create" | ||||||
|  |         type: string | ||||||
|  |         label: PostgreSQL Database | ||||||
|  |       - variable: postgresql.postgresqlUsername | ||||||
|  |         default: "passbook" | ||||||
|  |         description: "Database user to create" | ||||||
|  |         type: string | ||||||
|  |         label: PostgreSQL User | ||||||
|  |       - variable: postgresql.postgresqlPassword | ||||||
|  |         default: "" | ||||||
|  |         description: "password will be auto-generated if not specified" | ||||||
|  |         type: password | ||||||
|  |         label: PostgreSQL Password | ||||||
|  |   - variable: externalDatabase.host | ||||||
|  |     default: "" | ||||||
|  |     description: "Host of the external database" | ||||||
|  |     type: string | ||||||
|  |     label: External Database Host | ||||||
|  |     show_if: "postgresql.enabled=false" | ||||||
|  |     group: "Database Settings" | ||||||
|  |   - variable: externalDatabase.user | ||||||
|  |     default: "" | ||||||
|  |     description: "Existing username in the external DB" | ||||||
|  |     type: string | ||||||
|  |     label: External Database username | ||||||
|  |     show_if: "postgresql.enabled=false" | ||||||
|  |     group: "Database Settings" | ||||||
|  |   - variable: externalDatabase.password | ||||||
|  |     default: "" | ||||||
|  |     description: "External database password" | ||||||
|  |     type: password | ||||||
|  |     label: External Database password | ||||||
|  |     show_if: "postgresql.enabled=false" | ||||||
|  |     group: "Database Settings" | ||||||
|  |   - variable: externalDatabase.database | ||||||
|  |     default: "" | ||||||
|  |     description: "Name of the existing database" | ||||||
|  |     type: string | ||||||
|  |     label: External Database | ||||||
|  |     show_if: "postgresql.enabled=false" | ||||||
|  |     group: "Database Settings" | ||||||
|  |   - variable: externalDatabase.port | ||||||
|  |     default: "3306" | ||||||
|  |     description: "External database port number" | ||||||
|  |     type: string | ||||||
|  |     label: External Database Port | ||||||
|  |     show_if: "postgresql.enabled=false" | ||||||
|  |     group: "Database Settings" | ||||||
|  |   - variable: postgresql.persistence.enabled | ||||||
|  |     default: false | ||||||
|  |     description: "Enable persistent volume for PostgreSQL" | ||||||
|  |     type: boolean | ||||||
|  |     required: true | ||||||
|  |     label: PostgreSQL Persistent Volume Enabled | ||||||
|  |     show_if: "postgresql.enabled=true" | ||||||
|  |     show_subquestion_if: true | ||||||
|  |     group: "Database Settings" | ||||||
|  |     subquestions: | ||||||
|  |       - variable: postgresql.master.persistence.size | ||||||
|  |         default: "8Gi" | ||||||
|  |         description: "PostgreSQL Persistent Volume Size" | ||||||
|  |         type: string | ||||||
|  |         label: PostgreSQL Volume Size | ||||||
|  |       - variable: postgresql.master.persistence.storageClass | ||||||
|  |         default: "" | ||||||
|  |         description: "If undefined or null, uses the default StorageClass. Default to null" | ||||||
|  |         type: storageclass | ||||||
|  |         label: Default StorageClass for PostgreSQL | ||||||
|  |       - variable: postgresql.master.persistence.existingClaim | ||||||
|  |         default: "" | ||||||
|  |         description: "If not empty, uses the specified existing PVC instead of creating new one" | ||||||
|  |         type: string | ||||||
|  |         label: Existing Persistent Volume Claim for PostgreSQL | ||||||
| @ -1,9 +1,12 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis | - name: rabbitmq | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|   version: 5.1.0 |   version: 4.3.2 | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|   version: 3.10.1 |   version: 3.10.1 | ||||||
| digest: sha256:04bd136761f070e94a2ff32ff48ff87f5e07fbd451e5fd7f65551e3bd4680e5e | - name: redis | ||||||
| generated: 2019-02-08T12:08:49.090666+01:00 |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  |   version: 5.1.0 | ||||||
|  | digest: sha256:8bf68bc928a2e3c0f05139635be05fa0840554c7bde4cecd624fac78fb5fa5a3 | ||||||
|  | generated: 2019-03-21T11:06:51.553379+01:00 | ||||||
|  | |||||||
| @ -1,7 +1,10 @@ | |||||||
| dependencies: | dependencies: | ||||||
| - name: redis | - name: rabbitmq | ||||||
|   version: 5.1.0 |   version: 4.3.2 | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
| - name: postgresql | - name: postgresql | ||||||
|   version: 3.10.1 |   version: 3.10.1 | ||||||
|   repository: https://kubernetes-charts.storage.googleapis.com/ |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | - name: redis | ||||||
|  |   version: 5.1.0 | ||||||
|  |   repository: https://kubernetes-charts.storage.googleapis.com/ | ||||||
|  | |||||||
| @ -15,14 +15,14 @@ data: | |||||||
|         port: '' |         port: '' | ||||||
|     log: |     log: | ||||||
|       level: |       level: | ||||||
|         console: DEBUG |         console: WARNING | ||||||
|         file: DEBUG |         file: WARNING | ||||||
|       file: /dev/null |       file: /dev/null | ||||||
|       syslog: |       syslog: | ||||||
|         host: 127.0.0.1 |         host: 127.0.0.1 | ||||||
|         port: 514 |         port: 514 | ||||||
|     email: |     email: | ||||||
|       host: localhost |       host: {{ .Values.config.email.host }} | ||||||
|       port: 25 |       port: 25 | ||||||
|       user: '' |       user: '' | ||||||
|       password: '' |       password: '' | ||||||
| @ -36,7 +36,8 @@ data: | |||||||
|     debug: false |     debug: false | ||||||
|     secure_proxy_header: |     secure_proxy_header: | ||||||
|       HTTP_X_FORWARDED_PROTO: https |       HTTP_X_FORWARDED_PROTO: https | ||||||
|     redis: ":{{ .Values.redis.password }}@{{ .Release.Name }}-redis-master" |     rabbitmq: "user:{{ .Values.rabbitmq.rabbitmq.password }}@{{ .Release.Name }}-rabbitmq" | ||||||
|  |     redis: ":{{ .Values.redis.password }}@{{ .Release.Name }}-redis-master/0" | ||||||
|     # Error reporting, sends stacktrace to sentry.services.beryju.org |     # Error reporting, sends stacktrace to sentry.services.beryju.org | ||||||
|     error_report_enabled: {{ .Values.config.error_reporting }} |     error_report_enabled: {{ .Values.config.error_reporting }} | ||||||
|  |  | ||||||
| @ -46,6 +47,7 @@ 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 }} | ||||||
| @ -123,6 +125,7 @@ data: | |||||||
|         - passbook.oauth_client.source_types.reddit |         - passbook.oauth_client.source_types.reddit | ||||||
|         - passbook.oauth_client.source_types.supervisr |         - passbook.oauth_client.source_types.supervisr | ||||||
|         - passbook.oauth_client.source_types.twitter |         - passbook.oauth_client.source_types.twitter | ||||||
|  |         - passbook.oauth_client.source_types.azure_ad | ||||||
|     saml_idp: |     saml_idp: | ||||||
|       signing: true |       signing: true | ||||||
|       autosubmit: false |       autosubmit: false | ||||||
| @ -131,9 +134,4 @@ data: | |||||||
|       # List of python packages with provider types to load. |       # List of python packages with provider types to load. | ||||||
|       types: |       types: | ||||||
|         - passbook.saml_idp.processors.generic |         - passbook.saml_idp.processors.generic | ||||||
|         - passbook.saml_idp.processors.aws |  | ||||||
|         - passbook.saml_idp.processors.gitlab |  | ||||||
|         - passbook.saml_idp.processors.nextcloud |  | ||||||
|         - passbook.saml_idp.processors.salesforce |         - passbook.saml_idp.processors.salesforce | ||||||
|         - passbook.saml_idp.processors.shibboleth |  | ||||||
|         - passbook.saml_idp.processors.wordpress_orange |  | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ spec: | |||||||
|       labels: |       labels: | ||||||
|         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 | ||||||
|     spec: |     spec: | ||||||
|       volumes: |       volumes: | ||||||
|         - name: config-volume |         - name: config-volume | ||||||
| @ -25,7 +26,7 @@ spec: | |||||||
|             name: {{ include "passbook.fullname" . }}-config |             name: {{ include "passbook.fullname" . }}-config | ||||||
|       containers: |       containers: | ||||||
|         - name: {{ .Chart.Name }} |         - name: {{ .Chart.Name }} | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|           imagePullPolicy: IfNotPresent |           imagePullPolicy: IfNotPresent | ||||||
|           command: ["/bin/sh","-c"] |           command: ["/bin/sh","-c"] | ||||||
|           args: ["./manage.py migrate && ./manage.py web"] |           args: ["./manage.py migrate && ./manage.py web"] | ||||||
|  | |||||||
| @ -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 | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ spec: | |||||||
|       labels: |       labels: | ||||||
|         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: worker | ||||||
|     spec: |     spec: | ||||||
|       volumes: |       volumes: | ||||||
|         - name: config-volume |         - name: config-volume | ||||||
| @ -25,7 +26,7 @@ spec: | |||||||
|             name: {{ include "passbook.fullname" . }}-config |             name: {{ include "passbook.fullname" . }}-config | ||||||
|       containers: |       containers: | ||||||
|         - name: {{ .Chart.Name }} |         - name: {{ .Chart.Name }} | ||||||
|           image: "docker.pkg.beryju.org/passbook:{{ .Values.image.tag }}" |           image: "docker.beryju.org/passbook/server:{{ .Values.image.tag }}" | ||||||
|           imagePullPolicy: IfNotPresent |           imagePullPolicy: IfNotPresent | ||||||
|           command: ["./manage.py", "worker"] |           command: ["./manage.py", "worker"] | ||||||
|           ports: |           ports: | ||||||
|  | |||||||
| @ -5,7 +5,7 @@ | |||||||
| replicaCount: 1 | replicaCount: 1 | ||||||
|  |  | ||||||
| image: | image: | ||||||
|   tag: 0.1.8-beta |   tag: 0.2.0-beta | ||||||
|  |  | ||||||
| nameOverride: "" | nameOverride: "" | ||||||
|  |  | ||||||
| @ -14,11 +14,17 @@ 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 | ||||||
|  |  | ||||||
|  | rabbitmq: | ||||||
|  |   rabbitmq: | ||||||
|  |     password: foo | ||||||
|  |  | ||||||
| service: | service: | ||||||
|   type: ClusterIP |   type: ClusterIP | ||||||
|   port: 80 |   port: 80 | ||||||
| @ -31,7 +37,6 @@ ingress: | |||||||
|   path: / |   path: / | ||||||
|   hosts: |   hosts: | ||||||
|     - passbook.k8s.local |     - passbook.k8s.local | ||||||
|     - kubernetes-healthcheck-host |  | ||||||
|   defaultHost: passbook.k8s.local |   defaultHost: passbook.k8s.local | ||||||
|   tls: [] |   tls: [] | ||||||
|   #  - secretName: chart-example-tls |   #  - secretName: chart-example-tls | ||||||
|  | |||||||
| @ -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,7 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| # Check if this file is a symlink, if so, read real base dir |  | ||||||
| BASE_DIR=$(dirname $(readlink -f ${BASH_SOURCE[0]})) |  | ||||||
|  |  | ||||||
| cd $BASE_DIR |  | ||||||
| PYTHONPATH="${BASE_DIR}/vendor/" python3 manage.py $@ |  | ||||||
| @ -1,2 +1,2 @@ | |||||||
| """passbook""" | """passbook""" | ||||||
| __version__ = '0.1.8-beta' | __version__ = '0.2.0-beta' | ||||||
|  | |||||||
| @ -1,2 +1,2 @@ | |||||||
| """passbook admin""" | """passbook admin""" | ||||||
| __version__ = '0.1.8-beta' | __version__ = '0.2.0-beta' | ||||||
|  | |||||||
							
								
								
									
										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 | ||||||
							
								
								
									
										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,46 +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:property-mappings' 'passbook_admin:property-mapping-create' 'passbook_admin:property-mapping-update' 'passbook_admin:property-mapping-delete' %}"> |  | ||||||
|         <a href="{% url 'passbook_admin:property-mappings' %}">{% trans 'Property Mappings' %}</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 %} |  | ||||||
| @ -152,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"> | ||||||
| @ -192,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 %} | ||||||
|  | |||||||
| @ -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 %} | ||||||
|  | |||||||
| @ -36,7 +36,7 @@ | |||||||
|         <tbody> |         <tbody> | ||||||
|             {% for property_mapping in object_list %} |             {% for property_mapping in object_list %} | ||||||
|             <tr> |             <tr> | ||||||
|                 <td>{{ property_mapping.name }} ({{ property_mapping.slug }})</td> |                 <td>{{ property_mapping.name }}</td> | ||||||
|                 <td>{{ property_mapping|verbose_name }}</td> |                 <td>{{ property_mapping|verbose_name }}</td> | ||||||
|                 <td> |                 <td> | ||||||
|                     <a class="btn btn-default btn-sm" |                     <a class="btn btn-default btn-sm" | ||||||
|  | |||||||
| @ -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,10 +2,20 @@ | |||||||
|  |  | ||||||
| {% load i18n %} | {% load i18n %} | ||||||
| {% load utils %} | {% load utils %} | ||||||
|  | {% load static %} | ||||||
|  |  | ||||||
| {% block head %} | {% block head %} | ||||||
| {{ block.super }} | {{ block.super }} | ||||||
| {{ form.media.css }} | {{ 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 %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| @ -19,6 +29,8 @@ | |||||||
|       <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 %} | ||||||
|  |  | ||||||
|  | |||||||
| @ -5,6 +5,8 @@ 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 passbook.lib.utils.template import render_to_string | ||||||
|  |  | ||||||
| register = template.Library() | register = template.Library() | ||||||
| LOGGER = getLogger(__name__) | LOGGER = getLogger(__name__) | ||||||
|  |  | ||||||
| @ -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,7 +1,7 @@ | |||||||
| """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, |                                   invitations, overview, policy, | ||||||
|                                   property_mapping, providers, sources, users) |                                   property_mapping, providers, sources, users) | ||||||
|  |  | ||||||
| @ -67,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.admin.mixins import AdminRequiredMixin | from passbook.admin.mixins import AdminRequiredMixin | ||||||
| from passbook.core import __version__ | 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()) | ||||||
| @ -25,4 +34,5 @@ class AdministrationOverviewView(AdminRequiredMixin, TemplateView): | |||||||
|         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['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) | ||||||
|  | |||||||
| @ -11,6 +11,7 @@ from django.views.generic.detail import DetailView | |||||||
| from passbook.admin.forms.policies import PolicyTestForm | 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.core.policies import PolicyEngine | ||||||
| from passbook.lib.utils.reflection import path_to_class | from passbook.lib.utils.reflection import path_to_class | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -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: | ||||||
|  | |||||||
| @ -1,2 +1,2 @@ | |||||||
| """passbook api""" | """passbook api""" | ||||||
| __version__ = '0.1.8-beta' | __version__ = '0.2.0-beta' | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								passbook/app_gw/.DS_Store
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								passbook/app_gw/.DS_Store
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								passbook/app_gw/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								passbook/app_gw/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | """passbook Application Security Gateway Header""" | ||||||
|  | __version__ = '0.2.0-beta' | ||||||
							
								
								
									
										5
									
								
								passbook/app_gw/admin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								passbook/app_gw/admin.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | """passbook Application Security Gateway model admin""" | ||||||
|  |  | ||||||
|  | from passbook.lib.admin import admin_autoregister | ||||||
|  |  | ||||||
|  | admin_autoregister('passbook_app_gw') | ||||||
							
								
								
									
										16
									
								
								passbook/app_gw/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								passbook/app_gw/apps.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | """passbook Application Security Gateway app""" | ||||||
|  | from importlib import import_module | ||||||
|  |  | ||||||
|  | from django.apps import AppConfig | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PassbookApplicationApplicationGatewayConfig(AppConfig): | ||||||
|  |     """passbook app_gw app""" | ||||||
|  |  | ||||||
|  |     name = 'passbook.app_gw' | ||||||
|  |     label = 'passbook_app_gw' | ||||||
|  |     verbose_name = 'passbook Application Security Gateway' | ||||||
|  |     mountpoint = 'app_gw/' | ||||||
|  |  | ||||||
|  |     def ready(self): | ||||||
|  |         import_module('passbook.app_gw.signals') | ||||||
							
								
								
									
										66
									
								
								passbook/app_gw/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								passbook/app_gw/forms.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | |||||||
|  | """passbook Application Security Gateway Forms""" | ||||||
|  | from urllib.parse import urlparse | ||||||
|  |  | ||||||
|  | from django import forms | ||||||
|  | from django.contrib.admin.widgets import FilteredSelectMultiple | ||||||
|  | from django.forms import ValidationError | ||||||
|  | from django.utils.translation import gettext as _ | ||||||
|  |  | ||||||
|  | from passbook.app_gw.models import ApplicationGatewayProvider, RewriteRule | ||||||
|  | from passbook.lib.fields import DynamicArrayField | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ApplicationGatewayProviderForm(forms.ModelForm): | ||||||
|  |     """Security Gateway Provider form""" | ||||||
|  |  | ||||||
|  |     def clean_server_name(self): | ||||||
|  |         """Check if server_name is in DB already, since | ||||||
|  |         Postgres ArrayField doesn't suppport keys.""" | ||||||
|  |         current = self.cleaned_data.get('server_name') | ||||||
|  |         if ApplicationGatewayProvider.objects \ | ||||||
|  |                 .filter(server_name__overlap=current) \ | ||||||
|  |                 .exclude(pk=self.instance.pk).exists(): | ||||||
|  |             raise ValidationError(_("Server Name already in use.")) | ||||||
|  |         return current | ||||||
|  |  | ||||||
|  |     def clean_upstream(self): | ||||||
|  |         """Check that upstream begins with http(s)""" | ||||||
|  |         for upstream in self.cleaned_data.get('upstream'): | ||||||
|  |             _parsed_url = urlparse(upstream) | ||||||
|  |  | ||||||
|  |             if _parsed_url.scheme not in ('http', 'https'): | ||||||
|  |                 raise ValidationError(_("URL Scheme must be either http or https")) | ||||||
|  |         return self.cleaned_data.get('upstream') | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         model = ApplicationGatewayProvider | ||||||
|  |         fields = ['server_name', 'upstream', 'enabled', 'authentication_header', | ||||||
|  |                   'default_content_type', 'upstream_ssl_verification', 'property_mappings'] | ||||||
|  |         widgets = { | ||||||
|  |             'authentication_header': forms.TextInput(), | ||||||
|  |             'default_content_type': forms.TextInput(), | ||||||
|  |             'property_mappings': FilteredSelectMultiple(_('Property Mappings'), False) | ||||||
|  |         } | ||||||
|  |         field_classes = { | ||||||
|  |             'server_name': DynamicArrayField, | ||||||
|  |             'upstream': DynamicArrayField | ||||||
|  |         } | ||||||
|  |         labels = { | ||||||
|  |             'upstream_ssl_verification': _('Verify upstream SSL Certificates?'), | ||||||
|  |             'property_mappings': _('Rewrite Rules') | ||||||
|  |         } | ||||||
|  |  | ||||||
|  | class RewriteRuleForm(forms.ModelForm): | ||||||
|  |     """Rewrite Rule Form""" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         model = RewriteRule | ||||||
|  |         fields = ['name', 'match', 'halt', 'replacement', 'redirect', 'conditions'] | ||||||
|  |         widgets = { | ||||||
|  |             'name': forms.TextInput(), | ||||||
|  |             'match': forms.TextInput(attrs={'data-is-monospace': True}), | ||||||
|  |             'replacement': forms.TextInput(attrs={'data-is-monospace': True}), | ||||||
|  |             'conditions': FilteredSelectMultiple(_('Conditions'), False) | ||||||
|  |         } | ||||||
							
								
								
									
										33
									
								
								passbook/app_gw/middleware.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								passbook/app_gw/middleware.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | """passbook app_gw middleware""" | ||||||
|  | from django.views.generic import RedirectView | ||||||
|  |  | ||||||
|  | from passbook.app_gw.proxy.handler import RequestHandler | ||||||
|  | from passbook.lib.config import CONFIG | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ApplicationGatewayMiddleware: | ||||||
|  |     """Check if request should be proxied or handeled normally""" | ||||||
|  |  | ||||||
|  |     _app_gw_cache = {} | ||||||
|  |  | ||||||
|  |     def __init__(self, get_response): | ||||||
|  |         self.get_response = get_response | ||||||
|  |  | ||||||
|  |     def __call__(self, request): | ||||||
|  |         # Rudimentary cache | ||||||
|  |         host_header = request.META.get('HTTP_HOST') | ||||||
|  |         if host_header not in self._app_gw_cache: | ||||||
|  |             self._app_gw_cache[host_header] = RequestHandler.find_app_gw_for_request(request) | ||||||
|  |         if self._app_gw_cache[host_header]: | ||||||
|  |             return self.dispatch(request, self._app_gw_cache[host_header]) | ||||||
|  |         return self.get_response(request) | ||||||
|  |  | ||||||
|  |     def dispatch(self, request, app_gw): | ||||||
|  |         """Build proxied request and pass to upstream""" | ||||||
|  |         handler = RequestHandler(app_gw, request) | ||||||
|  |  | ||||||
|  |         if not handler.check_permission(): | ||||||
|  |             to_url = 'https://%s/?next=%s' % (CONFIG.get('domains')[0], request.get_full_path()) | ||||||
|  |             return RedirectView.as_view(url=to_url)(request) | ||||||
|  |  | ||||||
|  |         return handler.get_response() | ||||||
							
								
								
									
										
											BIN
										
									
								
								passbook/app_gw/migrations/.DS_Store
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								passbook/app_gw/migrations/.DS_Store
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										50
									
								
								passbook/app_gw/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								passbook/app_gw/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | |||||||
|  | # Generated by Django 2.1.7 on 2019-03-20 21:38 | ||||||
|  |  | ||||||
|  | import django.contrib.postgres.fields | ||||||
|  | import django.db.models.deletion | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     initial = True | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('passbook_core', '0020_groupmembershippolicy'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='ApplicationGatewayProvider', | ||||||
|  |             fields=[ | ||||||
|  |                 ('provider_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.Provider')), | ||||||
|  |                 ('server_name', django.contrib.postgres.fields.ArrayField(base_field=models.TextField(), size=None)), | ||||||
|  |                 ('upstream', django.contrib.postgres.fields.ArrayField(base_field=models.TextField(), size=None)), | ||||||
|  |                 ('enabled', models.BooleanField(default=True)), | ||||||
|  |                 ('authentication_header', models.TextField(default='X-Remote-User')), | ||||||
|  |                 ('default_content_type', models.TextField(default='application/octet-stream')), | ||||||
|  |                 ('upstream_ssl_verification', models.BooleanField(default=True)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Application Gateway Provider', | ||||||
|  |                 'verbose_name_plural': 'Application Gateway Providers', | ||||||
|  |             }, | ||||||
|  |             bases=('passbook_core.provider',), | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='RewriteRule', | ||||||
|  |             fields=[ | ||||||
|  |                 ('propertymapping_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='passbook_core.PropertyMapping')), | ||||||
|  |                 ('match', models.TextField()), | ||||||
|  |                 ('halt', models.BooleanField(default=False)), | ||||||
|  |                 ('replacement', models.TextField()), | ||||||
|  |                 ('redirect', models.CharField(choices=[('internal', 'Internal'), (301, 'Moved Permanently'), (302, 'Found')], max_length=50)), | ||||||
|  |                 ('conditions', models.ManyToManyField(to='passbook_core.Policy')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'verbose_name': 'Rewrite Rule', | ||||||
|  |                 'verbose_name_plural': 'Rewrite Rules', | ||||||
|  |             }, | ||||||
|  |             bases=('passbook_core.propertymapping',), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										18
									
								
								passbook/app_gw/migrations/0002_auto_20190321_1521.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								passbook/app_gw/migrations/0002_auto_20190321_1521.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | # Generated by Django 2.1.7 on 2019-03-21 15:21 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('passbook_app_gw', '0001_initial'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='rewriterule', | ||||||
|  |             name='conditions', | ||||||
|  |             field=models.ManyToManyField(blank=True, to='passbook_core.Policy'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										18
									
								
								passbook/app_gw/migrations/0003_auto_20190411_1314.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								passbook/app_gw/migrations/0003_auto_20190411_1314.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | # Generated by Django 2.2 on 2019-04-11 13:14 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('passbook_app_gw', '0002_auto_20190321_1521'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='applicationgatewayprovider', | ||||||
|  |             name='authentication_header', | ||||||
|  |             field=models.TextField(blank=True, default='X-Remote-User'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										0
									
								
								passbook/app_gw/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								passbook/app_gw/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										74
									
								
								passbook/app_gw/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								passbook/app_gw/models.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,74 @@ | |||||||
|  | """passbook app_gw models""" | ||||||
|  | import re | ||||||
|  |  | ||||||
|  | from django.contrib.postgres.fields import ArrayField | ||||||
|  | from django.db import models | ||||||
|  | from django.utils.translation import gettext as _ | ||||||
|  |  | ||||||
|  | from passbook.core.models import Policy, PropertyMapping, Provider | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ApplicationGatewayProvider(Provider): | ||||||
|  |     """Virtual server which proxies requests to any hostname in server_name to upstream""" | ||||||
|  |  | ||||||
|  |     server_name = ArrayField(models.TextField()) | ||||||
|  |     upstream = ArrayField(models.TextField()) | ||||||
|  |     enabled = models.BooleanField(default=True) | ||||||
|  |  | ||||||
|  |     authentication_header = models.TextField(default='X-Remote-User', blank=True) | ||||||
|  |     default_content_type = models.TextField(default='application/octet-stream') | ||||||
|  |     upstream_ssl_verification = models.BooleanField(default=True) | ||||||
|  |  | ||||||
|  |     form = 'passbook.app_gw.forms.ApplicationGatewayProviderForm' | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def name(self): | ||||||
|  |         """since this model has no name property, return a joined list of server_names as name""" | ||||||
|  |         return ', '.join(self.server_name) | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return "Application Gateway %s" % ', '.join(self.server_name) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         verbose_name = _('Application Gateway Provider') | ||||||
|  |         verbose_name_plural = _('Application Gateway Providers') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RewriteRule(PropertyMapping): | ||||||
|  |     """Rewrite requests matching `match` with `replacement`, if all polcies in `conditions` apply""" | ||||||
|  |  | ||||||
|  |     REDIRECT_INTERNAL = 'internal' | ||||||
|  |     REDIRECT_PERMANENT = 301 | ||||||
|  |     REDIRECT_FOUND = 302 | ||||||
|  |  | ||||||
|  |     REDIRECTS = ( | ||||||
|  |         (REDIRECT_INTERNAL, _('Internal')), | ||||||
|  |         (REDIRECT_PERMANENT, _('Moved Permanently')), | ||||||
|  |         (REDIRECT_FOUND, _('Found')), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     match = models.TextField() | ||||||
|  |     halt = models.BooleanField(default=False) | ||||||
|  |     conditions = models.ManyToManyField(Policy, blank=True) | ||||||
|  |     replacement = models.TextField() # python formatted strings, use {match.1} | ||||||
|  |     redirect = models.CharField(max_length=50, choices=REDIRECTS) | ||||||
|  |  | ||||||
|  |     form = 'passbook.app_gw.forms.RewriteRuleForm' | ||||||
|  |  | ||||||
|  |     _matcher = None | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def compiled_matcher(self): | ||||||
|  |         """Cache the compiled regex in memory""" | ||||||
|  |         if not self._matcher: | ||||||
|  |             self._matcher = re.compile(self.match) | ||||||
|  |         return self._matcher | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return "Rewrite Rule %s" % self.name | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |  | ||||||
|  |         verbose_name = _('Rewrite Rule') | ||||||
|  |         verbose_name_plural = _('Rewrite Rules') | ||||||
							
								
								
									
										0
									
								
								passbook/app_gw/proxy/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								passbook/app_gw/proxy/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										8
									
								
								passbook/app_gw/proxy/exceptions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								passbook/app_gw/proxy/exceptions.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | """Exception classes""" | ||||||
|  |  | ||||||
|  | class ReverseProxyException(Exception): | ||||||
|  |     """Base for revproxy exception""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InvalidUpstream(ReverseProxyException): | ||||||
|  |     """Invalid upstream set""" | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	