Compare commits
13 Commits
version/0.
...
version/0.
| Author | SHA1 | Date | |
|---|---|---|---|
| a53a269a8c | |||
| 59565a5286 | |||
| ae3c092238 | |||
| e98e5e4e3e | |||
| d50c7ec8d4 | |||
| c0fdf377d1 | |||
| 70c11c8988 | |||
| 67b19becc1 | |||
| ae64024ef4 | |||
| e6571826cb | |||
| c621e61978 | |||
| 3626fa4b98 | |||
| 01b0eb159a |
@ -1,5 +1,5 @@
|
|||||||
[bumpversion]
|
[bumpversion]
|
||||||
current_version = 0.1.17-beta
|
current_version = 0.1.20-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>.*)
|
||||||
@ -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]
|
||||||
|
|||||||
@ -16,10 +16,10 @@ services:
|
|||||||
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:
|
include:
|
||||||
- /allauth/.gitlab-ci.yml
|
- /client-packages/allauth/.gitlab-ci.yml
|
||||||
|
|
||||||
isort:
|
isort:
|
||||||
script:
|
script:
|
||||||
@ -54,7 +54,7 @@ package-docker:
|
|||||||
before_script:
|
before_script:
|
||||||
- echo "{\"auths\":{\"docker.$NEXUS_URL\":{\"auth\":\"$NEXUS_AUTH\"}}}" > /kaniko/.docker/config.json
|
- echo "{\"auths\":{\"docker.$NEXUS_URL\":{\"auth\":\"$NEXUS_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.17-beta
|
- /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.20-beta
|
||||||
stage: build
|
stage: build
|
||||||
only:
|
only:
|
||||||
- tags
|
- tags
|
||||||
@ -94,6 +94,32 @@ package-debian:
|
|||||||
- tags
|
- tags
|
||||||
- /^version/.*$/
|
- /^version/.*$/
|
||||||
|
|
||||||
|
package-client-package-allauth:
|
||||||
|
script:
|
||||||
|
- cd client-packages/allauth
|
||||||
|
- python setup.py sdist
|
||||||
|
- twine upload --username $TWINE_USERNAME --password $TWINE_PASSWORD dist/*
|
||||||
|
stage: build
|
||||||
|
only:
|
||||||
|
refs:
|
||||||
|
- tags
|
||||||
|
- /^version/.*$/
|
||||||
|
changes:
|
||||||
|
- client-packages/allauth/**
|
||||||
|
|
||||||
|
package-client-package-sentry:
|
||||||
|
script:
|
||||||
|
- cd client-packages/sentry-auth-passbook
|
||||||
|
- python setup.py sdist
|
||||||
|
- twine upload --username $TWINE_USERNAME --password $TWINE_PASSWORD dist/*
|
||||||
|
stage: build
|
||||||
|
only:
|
||||||
|
refs:
|
||||||
|
- tags
|
||||||
|
- /^version/.*$/
|
||||||
|
changes:
|
||||||
|
- client-packages/sentry-auth-passbook/**
|
||||||
|
|
||||||
# docs:
|
# docs:
|
||||||
# stage: docs
|
# stage: docs
|
||||||
# only:
|
# only:
|
||||||
|
|||||||
@ -7,6 +7,7 @@ ignore-paths:
|
|||||||
- migrations
|
- migrations
|
||||||
- docs
|
- docs
|
||||||
- node_modules
|
- node_modules
|
||||||
|
- client-packages
|
||||||
|
|
||||||
uses:
|
uses:
|
||||||
- django
|
- django
|
||||||
|
|||||||
@ -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.1.20-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.1.20-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
|
||||||
21
debian/changelog
vendored
21
debian/changelog
vendored
@ -1,3 +1,24 @@
|
|||||||
|
passbook (0.1.20) stable; urgency=medium
|
||||||
|
|
||||||
|
* bump version: 0.1.18-beta -> 0.1.19-beta
|
||||||
|
* fix GitHub Pretend again
|
||||||
|
* add user settings for Sources
|
||||||
|
|
||||||
|
-- Jens Langhammer <jens.langhammer@beryju.org> Wed, 13 Mar 2019 15:49:44 +0000
|
||||||
|
|
||||||
|
passbook (0.1.18) stable; urgency=medium
|
||||||
|
|
||||||
|
* bump version: 0.1.16-beta -> 0.1.17-beta
|
||||||
|
* fix Server Error when downloading metadata
|
||||||
|
* add sentry client
|
||||||
|
* fix included yaml file
|
||||||
|
* adjust versions for client packages, auto build client-packages
|
||||||
|
* bump version: 0.1.17-beta -> 0.1.18-beta
|
||||||
|
* fix API Call for sentry-client, add missing template
|
||||||
|
* fix GitHub Pretend throwing a 500 error
|
||||||
|
|
||||||
|
-- Jens Langhammer <jens.langhammer@beryju.org> Wed, 13 Mar 2019 14:14:10 +0000
|
||||||
|
|
||||||
passbook (0.1.17) stable; urgency=medium
|
passbook (0.1.17) stable; urgency=medium
|
||||||
|
|
||||||
* bump version: 0.1.15-beta -> 0.1.16-beta
|
* bump version: 0.1.15-beta -> 0.1.16-beta
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
appVersion: "0.1.17-beta"
|
appVersion: "0.1.20-beta"
|
||||||
description: A Helm chart for passbook.
|
description: A Helm chart for passbook.
|
||||||
name: passbook
|
name: passbook
|
||||||
version: "0.1.17-beta"
|
version: "0.1.20-beta"
|
||||||
icon: https://passbook.beryju.org/images/logo.png
|
icon: https://passbook.beryju.org/images/logo.png
|
||||||
|
|||||||
@ -5,7 +5,7 @@
|
|||||||
replicaCount: 1
|
replicaCount: 1
|
||||||
|
|
||||||
image:
|
image:
|
||||||
tag: 0.1.17-beta
|
tag: 0.1.20-beta
|
||||||
|
|
||||||
nameOverride: ""
|
nameOverride: ""
|
||||||
|
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook"""
|
"""passbook"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook admin"""
|
"""passbook admin"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook api"""
|
"""passbook api"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook audit Header"""
|
"""passbook audit Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook captcha_factor Header"""
|
"""passbook captcha_factor Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook core"""
|
"""passbook core"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -186,6 +186,12 @@ class Source(PolicyModel):
|
|||||||
"""Return additional Info, such as a callback URL. Show in the administration interface."""
|
"""Return additional Info, such as a callback URL. Show in the administration interface."""
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def has_user_settings(self):
|
||||||
|
"""Entrypoint to integrate with User settings. Can either return False if no
|
||||||
|
user settings are available, or a tuple or string, string, string where the first string
|
||||||
|
is the name the item has, the second string is the icon and the third is the view-name."""
|
||||||
|
return False
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
|
|||||||
@ -26,7 +26,7 @@
|
|||||||
<li class="dropdown">
|
<li class="dropdown">
|
||||||
<button class="btn btn-link dropdown-toggle nav-item-iconic" id="dropdownMenu1" data-toggle="dropdown"
|
<button class="btn btn-link dropdown-toggle nav-item-iconic" id="dropdownMenu1" data-toggle="dropdown"
|
||||||
aria-haspopup="true" aria-expanded="true">
|
aria-haspopup="true" aria-expanded="true">
|
||||||
<span title="Help" class="fa pficon-help dropdown-title"></span>
|
<span title="Help" class="fa pficon-help"></span>
|
||||||
</button>
|
</button>
|
||||||
<ul class="dropdown-menu" aria-labelledby="dropdownMenu1">
|
<ul class="dropdown-menu" aria-labelledby="dropdownMenu1">
|
||||||
{% comment %} <li><a href="#0">Help</a></li> {% endcomment %}
|
{% comment %} <li><a href="#0">Help</a></li> {% endcomment %}
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% load is_active %}
|
{% load is_active %}
|
||||||
|
{% load static %}
|
||||||
{% load passbook_user_settings %}
|
{% load passbook_user_settings %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
@ -24,6 +25,15 @@
|
|||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
<li class="nav-divider"></li>
|
||||||
|
{% user_sources as us %}
|
||||||
|
{% for name, icon, link in us %}
|
||||||
|
<li class="{% if link == request.get_full_path %} active {% endif %}">
|
||||||
|
<a href="{{ link }}">
|
||||||
|
<img src="{% static icon %}" alt=""> {{ name }}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
from django import template
|
from django import template
|
||||||
|
|
||||||
from passbook.core.models import Factor
|
from passbook.core.models import Factor, Source
|
||||||
from passbook.core.policies import PolicyEngine
|
from passbook.core.policies import PolicyEngine
|
||||||
|
|
||||||
register = template.Library()
|
register = template.Library()
|
||||||
@ -20,3 +20,17 @@ def user_factors(context):
|
|||||||
if policy_engine.passing and _link:
|
if policy_engine.passing and _link:
|
||||||
matching_factors.append(_link)
|
matching_factors.append(_link)
|
||||||
return matching_factors
|
return matching_factors
|
||||||
|
|
||||||
|
@register.simple_tag(takes_context=True)
|
||||||
|
def user_sources(context):
|
||||||
|
"""Return a list of all sources which are enabled for the user"""
|
||||||
|
user = context.get('request').user
|
||||||
|
_all_sources = Source.objects.filter(enabled=True).select_subclasses()
|
||||||
|
matching_sources = []
|
||||||
|
for factor in _all_sources:
|
||||||
|
_link = factor.has_user_settings()
|
||||||
|
policy_engine = PolicyEngine(factor.policies.all())
|
||||||
|
policy_engine.for_user(user).with_request(context.get('request')).build()
|
||||||
|
if policy_engine.passing and _link:
|
||||||
|
matching_sources.append(_link)
|
||||||
|
return matching_sources
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook hibp_policy"""
|
"""passbook hibp_policy"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""Passbook ldap app Header"""
|
"""Passbook ldap app Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook lib"""
|
"""passbook lib"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook oauth_client Header"""
|
"""passbook oauth_client Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,7 +1,7 @@
|
|||||||
"""OAuth Client models"""
|
"""OAuth Client models"""
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.urls import reverse_lazy
|
from django.urls import reverse, reverse_lazy
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
from passbook.core.models import Source, UserSourceConnection
|
from passbook.core.models import Source, UserSourceConnection
|
||||||
@ -38,6 +38,16 @@ class OAuthSource(Source):
|
|||||||
return "Callback URL: '%s'" % reverse_lazy('passbook_oauth_client:oauth-client-callback',
|
return "Callback URL: '%s'" % reverse_lazy('passbook_oauth_client:oauth-client-callback',
|
||||||
kwargs={'source_slug': self.slug})
|
kwargs={'source_slug': self.slug})
|
||||||
|
|
||||||
|
def has_user_settings(self):
|
||||||
|
"""Entrypoint to integrate with User settings. Can either return False if no
|
||||||
|
user settings are available, or a tuple or string, string, string where the first string
|
||||||
|
is the name the item has, the second string is the icon and the third is the view-name."""
|
||||||
|
icon = 'img/%s.svg' % self.get_login_button[1]
|
||||||
|
view_name = 'passbook_oauth_client:oauth-client-user'
|
||||||
|
return self.name, icon, reverse((view_name), kwargs={
|
||||||
|
'source_slug': self.slug
|
||||||
|
})
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
|
|
||||||
verbose_name = _('Generic OAuth Source')
|
verbose_name = _('Generic OAuth Source')
|
||||||
|
|||||||
@ -1,6 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
|
|
||||||
{% any_provider as enabled %}
|
|
||||||
{% if enabled %}
|
|
||||||
<div class="btn-group btn-primary btn-block">
|
|
||||||
{% endif %}
|
|
||||||
@ -1,6 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
|
|
||||||
{% provider_exists 'facebook' as facebook_enabled %}
|
|
||||||
{% if facebook_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='facebook' %}" class="btn" style="background-color:#4267b2;color:white;margin-top:10px;width:100%;"><i class="fa fa-facebook-official" aria-hidden="true"></i></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,6 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
|
|
||||||
{% provider_exists 'twitter' as twitter_enabled %}
|
|
||||||
{% if twitter_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='twitter' %}" class="btn" style="background-color:#55ACEE;color:white;margin-top:10px;width:100%;"><i class="fa fa-twitter" aria-hidden="true"></i></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,7 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
{% load static %}
|
|
||||||
|
|
||||||
{% provider_exists 'google' as google_enabled %}
|
|
||||||
{% if google_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='google' %}" class="btn" style="background-color:white;color:black;margin-top:10px;width:100%;"><img src="{% static 'img/google.svg' %}" style="height:12px"></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,6 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
|
|
||||||
{% provider_exists 'github' as github_enabled %}
|
|
||||||
{% if github_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='github' %}" class="btn" style="background-color:#444444;color:white;margin-top:10px;width:100%;"><i class="fa fa-github" aria-hidden="true"></i></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,7 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
{% load static %}
|
|
||||||
|
|
||||||
{% provider_exists 'discord' as discord_enabled %}
|
|
||||||
{% if discord_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='discord' %}" class="btn" style="background-color:#2C2F33;color:white;margin-top:10px;width:100%;"><img src="{% static 'img/discord.svg' %}" style="height:12px"></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,7 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
{% load static %}
|
|
||||||
|
|
||||||
{% provider_exists 'reddit' as reddit_enabled %}
|
|
||||||
{% if reddit_enabled %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider='reddit' %}" class="btn" style="background-color:#ff4500;color:white;margin-top:10px;width:100%;"><img src="{% static 'img/reddit.svg' %}" style="height:20px;margin-top:-5px;"></a>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,6 +0,0 @@
|
|||||||
{% load passbook_oauth_client %}
|
|
||||||
|
|
||||||
{% any_provider as enabled %}
|
|
||||||
{% if enabled %}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
@ -1,54 +0,0 @@
|
|||||||
{% extends "user/base.html" %}
|
|
||||||
|
|
||||||
{% load utils %}
|
|
||||||
{% load i18n %}
|
|
||||||
|
|
||||||
{% block title %}
|
|
||||||
{% title "Overview" %}
|
|
||||||
{% endblock %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<h1><clr-icon shape="connect" size="48"></clr-icon>{% trans "OAuth2" %}</h1>
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-md-12">
|
|
||||||
<div class="card">
|
|
||||||
<div class="card-header">
|
|
||||||
{% trans "Connected Accounts" %}
|
|
||||||
</div>
|
|
||||||
<div class="card-footer">
|
|
||||||
{% if provider_state %}
|
|
||||||
<table class="table">
|
|
||||||
<thead>
|
|
||||||
<th>
|
|
||||||
<th>{% trans 'Provider' %}</th>
|
|
||||||
<th>{% trans 'Status' %}</th>
|
|
||||||
<th>{% trans 'Action' %}</th>
|
|
||||||
<th>{% trans 'ID' %}</th>
|
|
||||||
</th>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{% for data in provider_state %}
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>{% trans data.provider.ui_name %}</td>
|
|
||||||
<td>{{ data.state|yesno:"Connected,Not Connected" }}</td>
|
|
||||||
<td>
|
|
||||||
{% if data.state == False %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-login' provider=data.provider.name %}">Connect</a>
|
|
||||||
{% else %}
|
|
||||||
<a href="{% url 'passbook_oauth_client:oauth-client-disconnect' provider=data.provider.name %}">Disconnect</a>
|
|
||||||
{% endif %}
|
|
||||||
</td>
|
|
||||||
<td>{{ data.aas.first.identifier }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
{% else %}
|
|
||||||
<p>{% trans "No Providers configured!" %}</p>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
||||||
18
passbook/oauth_client/templates/oauth_client/user.html
Normal file
18
passbook/oauth_client/templates/oauth_client/user.html
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
{% extends "user/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block page %}
|
||||||
|
<h1>{{ source.name }}</h1>
|
||||||
|
{% if connections.exists %}
|
||||||
|
<p>{% trans 'Connected.' %}</p>
|
||||||
|
<a class="btn btn-danger" href="{% url 'passbook_oauth_client:oauth-client-disconnect' source_slug=source.slug %}">
|
||||||
|
{% trans 'Disconnect' %}
|
||||||
|
</a>
|
||||||
|
{% else %}
|
||||||
|
<p>Not connected.</p>
|
||||||
|
<a class="btn btn-primary" href="{% url 'passbook_oauth_client:oauth-client-login' source_slug=source.slug %}">
|
||||||
|
{% trans 'Connect' %}
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
{% endblock %}
|
||||||
@ -3,7 +3,7 @@
|
|||||||
from django.urls import path
|
from django.urls import path
|
||||||
|
|
||||||
from passbook.oauth_client.source_types.manager import RequestKind
|
from passbook.oauth_client.source_types.manager import RequestKind
|
||||||
from passbook.oauth_client.views import core, dispatcher
|
from passbook.oauth_client.views import core, dispatcher, user
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path('login/<slug:source_slug>/', dispatcher.DispatcherView.as_view(
|
path('login/<slug:source_slug>/', dispatcher.DispatcherView.as_view(
|
||||||
@ -12,4 +12,6 @@ urlpatterns = [
|
|||||||
kind=RequestKind.callback), name='oauth-client-callback'),
|
kind=RequestKind.callback), name='oauth-client-callback'),
|
||||||
path('disconnect/<slug:source_slug>/', core.DisconnectView.as_view(),
|
path('disconnect/<slug:source_slug>/', core.DisconnectView.as_view(),
|
||||||
name='oauth-client-disconnect'),
|
name='oauth-client-disconnect'),
|
||||||
|
path('user/<slug:source_slug>/', user.UserSettingsView.as_view(),
|
||||||
|
name='oauth-client-user'),
|
||||||
]
|
]
|
||||||
|
|||||||
20
passbook/oauth_client/views/user.py
Normal file
20
passbook/oauth_client/views/user.py
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
"""passbook oauth_client user views"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django.views.generic import TemplateView
|
||||||
|
|
||||||
|
from passbook.oauth_client.models import OAuthSource, UserOAuthSourceConnection
|
||||||
|
|
||||||
|
|
||||||
|
class UserSettingsView(LoginRequiredMixin, TemplateView):
|
||||||
|
"""Show user current connection state"""
|
||||||
|
|
||||||
|
template_name = 'oauth_client/user.html'
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
source = get_object_or_404(OAuthSource, slug=self.kwargs.get('source_slug'))
|
||||||
|
connections = UserOAuthSourceConnection.objects.filter(user=self.request.user,
|
||||||
|
source=source)
|
||||||
|
kwargs['source'] = source
|
||||||
|
kwargs['connections'] = connections
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook oauth_provider Header"""
|
"""passbook oauth_provider Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook otp Header"""
|
"""passbook otp Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook password_expiry"""
|
"""passbook password_expiry"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -1,16 +1,24 @@
|
|||||||
"""passbook pretend GitHub Views"""
|
"""passbook pretend GitHub Views"""
|
||||||
from django.http import JsonResponse
|
from django.http import JsonResponse
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
from django.views import View
|
from django.views import View
|
||||||
|
from oauth2_provider.models import AccessToken
|
||||||
|
|
||||||
|
|
||||||
class GitHubUserView(View):
|
class GitHubUserView(View):
|
||||||
"""Emulate GitHub's /user API Endpoint"""
|
"""Emulate GitHub's /user API Endpoint"""
|
||||||
|
|
||||||
|
def verify_access_token(self):
|
||||||
|
"""Verify access token manually since github uses /user?access_token=..."""
|
||||||
|
token = get_object_or_404(AccessToken, token=self.request.GET.get('access_token', ''))
|
||||||
|
return token.user
|
||||||
|
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
"""Emulate GitHub's /user API Endpoint"""
|
"""Emulate GitHub's /user API Endpoint"""
|
||||||
|
user = self.verify_access_token()
|
||||||
return JsonResponse({
|
return JsonResponse({
|
||||||
"login": request.user.username,
|
"login": user.username,
|
||||||
"id": request.user.pk,
|
"id": user.pk,
|
||||||
"node_id": "",
|
"node_id": "",
|
||||||
"avatar_url": "",
|
"avatar_url": "",
|
||||||
"gravatar_id": "",
|
"gravatar_id": "",
|
||||||
@ -27,19 +35,19 @@ class GitHubUserView(View):
|
|||||||
"received_events_url": "",
|
"received_events_url": "",
|
||||||
"type": "User",
|
"type": "User",
|
||||||
"site_admin": False,
|
"site_admin": False,
|
||||||
"name": request.user.name,
|
"name": user.name,
|
||||||
"company": "",
|
"company": "",
|
||||||
"blog": "",
|
"blog": "",
|
||||||
"location": "",
|
"location": "",
|
||||||
"email": request.user.email,
|
"email": user.email,
|
||||||
"hireable": False,
|
"hireable": False,
|
||||||
"bio": "",
|
"bio": "",
|
||||||
"public_repos": 0,
|
"public_repos": 0,
|
||||||
"public_gists": 0,
|
"public_gists": 0,
|
||||||
"followers": 0,
|
"followers": 0,
|
||||||
"following": 0,
|
"following": 0,
|
||||||
"created_at": request.user.date_joined,
|
"created_at": user.date_joined,
|
||||||
"updated_at": request.user.date_joined,
|
"updated_at": user.date_joined,
|
||||||
"private_gists": 0,
|
"private_gists": 0,
|
||||||
"total_private_repos": 0,
|
"total_private_repos": 0,
|
||||||
"owned_private_repos": 0,
|
"owned_private_repos": 0,
|
||||||
|
|||||||
@ -1,2 +1,2 @@
|
|||||||
"""passbook saml_idp Header"""
|
"""passbook saml_idp Header"""
|
||||||
__version__ = '0.1.17-beta'
|
__version__ = '0.1.20-beta'
|
||||||
|
|||||||
@ -7,6 +7,4 @@ class GitLabProcessor(Processor):
|
|||||||
"""GitLab Response Handler Processor for testing against django-saml2-sp."""
|
"""GitLab Response Handler Processor for testing against django-saml2-sp."""
|
||||||
|
|
||||||
def _determine_audience(self):
|
def _determine_audience(self):
|
||||||
# Nextcloud expects an audience in this format
|
|
||||||
# https://<host>
|
|
||||||
self._audience = self._remote.acs_url.replace('/users/auth/saml/callback', '')
|
self._audience = self._remote.acs_url.replace('/users/auth/saml/callback', '')
|
||||||
|
|||||||
@ -206,7 +206,9 @@ class DescriptorDownloadView(AccessRequiredView):
|
|||||||
def get(self, request, application):
|
def get(self, request, application):
|
||||||
"""Replies with the XML Metadata IDSSODescriptor."""
|
"""Replies with the XML Metadata IDSSODescriptor."""
|
||||||
entity_id = CONFIG.y('saml_idp.issuer')
|
entity_id = CONFIG.y('saml_idp.issuer')
|
||||||
slo_url = request.build_absolute_uri(reverse('passbook_saml_idp:saml-logout'))
|
slo_url = request.build_absolute_uri(reverse('passbook_saml_idp:saml-logout', kwargs={
|
||||||
|
'application': application
|
||||||
|
}))
|
||||||
sso_url = request.build_absolute_uri(reverse('passbook_saml_idp:saml-login', kwargs={
|
sso_url = request.build_absolute_uri(reverse('passbook_saml_idp:saml-login', kwargs={
|
||||||
'application': application
|
'application': application
|
||||||
}))
|
}))
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
-r requirements.txt
|
-r requirements.txt
|
||||||
-r allauth/requirements.txt
|
-r client-packages/allauth/requirements.txt
|
||||||
coverage
|
coverage
|
||||||
isort
|
isort
|
||||||
astroid==2.0.4
|
astroid==2.0.4
|
||||||
@ -13,3 +13,4 @@ unittest-xml-reporting
|
|||||||
autopep8
|
autopep8
|
||||||
bandit
|
bandit
|
||||||
bumpversion
|
bumpversion
|
||||||
|
twine
|
||||||
|
|||||||
Reference in New Issue
Block a user