2016-08-13 00:49:53 +08:00
import base64
2015-12-13 16:34:12 +07:00
import json
2018-04-12 11:18:44 +07:00
import logging as logger
2016-08-13 00:49:53 +08:00
import os
2015-12-13 16:34:12 +07:00
import traceback
2016-11-21 15:52:07 +01:00
import re
2016-08-13 00:49:53 +08:00
from distutils . util import strtobool
from distutils . version import StrictVersion
2015-12-13 16:34:12 +07:00
from functools import wraps
2016-08-13 00:49:53 +08:00
from io import BytesIO
import jinja2
2016-09-17 06:41:22 -07:00
import qrcode as qrc
import qrcode . image . svg as qrc_svg
2018-01-22 18:22:19 +03:00
from flask import g , request , make_response , jsonify , render_template , session , redirect , url_for , send_from_directory , abort , flash
2016-07-01 21:41:41 +02:00
from flask_login import login_user , logout_user , current_user , login_required
2015-12-17 00:50:28 +07:00
from werkzeug import secure_filename
2016-08-13 00:49:53 +08:00
from werkzeug . security import gen_salt
2015-12-13 16:34:12 +07:00
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
from . models import User , Account , Domain , Record , Server , History , Anonymous , Setting , DomainSetting , DomainTemplate , DomainTemplateRecord
2017-09-22 15:28:09 +01:00
from app import app , login_manager , github , google
2018-03-30 13:49:35 +07:00
from app . lib import utils
2018-06-07 09:28:14 +07:00
from app . decorators import admin_role_required , can_access_domain , can_configure_dnssec
2016-06-16 15:33:05 +07:00
2018-03-28 11:43:54 +02:00
if app . config [ ' SAML_ENABLED ' ] :
from onelogin . saml2 . auth import OneLogin_Saml2_Auth
from onelogin . saml2 . utils import OneLogin_Saml2_Utils
2015-12-13 16:34:12 +07:00
2018-04-12 11:18:44 +07:00
logging = logger . getLogger ( __name__ )
2015-12-13 16:34:12 +07:00
2018-03-31 08:21:02 +07:00
# FILTERS
2018-06-10 15:16:28 +02:00
app . jinja_env . filters [ ' display_record_name ' ] = utils . display_record_name
app . jinja_env . filters [ ' display_master_name ' ] = utils . display_master_name
app . jinja_env . filters [ ' display_second_to_time ' ] = utils . display_time
app . jinja_env . filters [ ' email_to_gravatar_url ' ] = utils . email_to_gravatar_url
2015-12-13 16:34:12 +07:00
2016-06-26 20:53:29 +07:00
# Flag for pdns v4.x.x
# TODO: Find another way to do this
PDNS_VERSION = app . config [ ' PDNS_VERSION ' ]
if StrictVersion ( PDNS_VERSION ) > = StrictVersion ( ' 4.0.0 ' ) :
NEW_SCHEMA = True
else :
NEW_SCHEMA = False
2018-03-30 13:49:35 +07:00
2016-04-29 15:36:37 -06:00
@app.context_processor
def inject_fullscreen_layout_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' fullscreen_layout ' )
return dict ( fullscreen_layout_setting = strtobool ( setting_value ) )
2016-04-29 15:36:37 -06:00
2018-03-30 13:49:35 +07:00
2016-05-15 12:47:02 -06:00
@app.context_processor
def inject_record_helper_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' record_helper ' )
return dict ( record_helper_setting = strtobool ( setting_value ) )
2016-05-15 12:47:02 -06:00
2018-03-30 13:49:35 +07:00
2016-07-05 15:28:02 +00:00
@app.context_processor
def inject_login_ldap_first_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' login_ldap_first ' )
return dict ( login_ldap_first_setting = strtobool ( setting_value ) )
2016-07-05 15:28:02 +00:00
2018-03-30 13:49:35 +07:00
2016-06-08 19:23:08 -06:00
@app.context_processor
def inject_default_record_table_size_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' default_record_table_size ' )
return dict ( default_record_table_size_setting = setting_value )
2016-06-08 19:23:08 -06:00
2018-03-30 13:49:35 +07:00
2016-07-05 15:14:41 +00:00
@app.context_processor
def inject_default_domain_table_size_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' default_domain_table_size ' )
return dict ( default_domain_table_size_setting = setting_value )
2016-07-05 15:14:41 +00:00
2018-03-30 13:49:35 +07:00
2016-11-21 13:42:00 +01:00
@app.context_processor
def inject_auto_ptr_setting ( ) :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
setting_value = Setting ( ) . get ( ' auto_ptr ' )
return dict ( auto_ptr_setting = strtobool ( setting_value ) )
2016-11-21 13:42:00 +01:00
2018-03-30 13:49:35 +07:00
2015-12-13 16:34:12 +07:00
# START USER AUTHENTICATION HANDLER
@app.before_request
def before_request ( ) :
# check site maintenance mode first
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
maintenance = Setting ( ) . get ( ' maintenance ' )
if strtobool ( maintenance ) :
2015-12-13 16:34:12 +07:00
return render_template ( ' maintenance.html ' )
# check if user is anonymous
g . user = current_user
login_manager . anonymous_user = Anonymous
2016-07-13 21:33:21 +07:00
2015-12-13 16:34:12 +07:00
@login_manager.user_loader
def load_user ( id ) :
"""
This will be current_user
"""
return User . query . get ( int ( id ) )
2016-06-20 16:32:14 +07:00
def dyndns_login_required ( f ) :
@wraps ( f )
def decorated_function ( * args , * * kwargs ) :
if current_user . is_authenticated is False :
return render_template ( ' dyndns.html ' , response = ' badauth ' ) , 200
return f ( * args , * * kwargs )
return decorated_function
2018-03-30 13:49:35 +07:00
2016-06-20 16:32:14 +07:00
@login_manager.request_loader
def login_via_authorization_header ( request ) :
auth_header = request . headers . get ( ' Authorization ' )
if auth_header :
auth_header = auth_header . replace ( ' Basic ' , ' ' , 1 )
try :
2018-04-10 07:08:22 +07:00
auth_header = str ( base64 . b64decode ( auth_header ) , ' utf-8 ' )
2016-06-20 16:32:14 +07:00
username , password = auth_header . split ( " : " )
2018-03-30 13:49:35 +07:00
except TypeError as e :
2016-06-20 16:32:14 +07:00
return None
user = User ( username = username , password = password , plain_text_password = password )
try :
auth = user . is_validate ( method = ' LOCAL ' )
if auth == False :
return None
else :
login_user ( user , remember = False )
return user
2018-03-30 13:49:35 +07:00
except :
2016-06-20 16:32:14 +07:00
return None
return None
2015-12-13 16:34:12 +07:00
# END USER AUTHENTICATION HANDLER
# START VIEWS
2016-05-15 16:01:57 -06:00
@app.errorhandler ( 400 )
def http_bad_request ( e ) :
return redirect ( url_for ( ' error ' , code = 400 ) )
2018-03-30 13:49:35 +07:00
2016-05-15 16:01:57 -06:00
@app.errorhandler ( 401 )
def http_unauthorized ( e ) :
return redirect ( url_for ( ' error ' , code = 401 ) )
2018-03-30 13:49:35 +07:00
2016-05-15 16:01:57 -06:00
@app.errorhandler ( 404 )
def http_internal_server_error ( e ) :
return redirect ( url_for ( ' error ' , code = 404 ) )
2018-03-30 13:49:35 +07:00
2016-05-15 16:01:57 -06:00
@app.errorhandler ( 500 )
def http_page_not_found ( e ) :
return redirect ( url_for ( ' error ' , code = 500 ) )
2018-03-30 13:49:35 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /error/<path:code> ' )
2015-12-13 16:34:12 +07:00
def error ( code , msg = None ) :
supported_code = ( ' 400 ' , ' 401 ' , ' 404 ' , ' 500 ' )
if code in supported_code :
2018-04-01 07:57:41 +07:00
return render_template ( ' errors/ {0} .html ' . format ( code ) , msg = msg ) , int ( code )
2015-12-13 16:34:12 +07:00
else :
2016-05-15 16:01:57 -06:00
return render_template ( ' errors/404.html ' ) , 404
2015-12-13 16:34:12 +07:00
2018-03-30 13:49:35 +07:00
2016-04-22 18:19:03 -06:00
@app.route ( ' /register ' , methods = [ ' GET ' ] )
def register ( ) :
2016-06-13 11:48:48 +07:00
SIGNUP_ENABLED = app . config [ ' SIGNUP_ENABLED ' ]
if SIGNUP_ENABLED :
return render_template ( ' register.html ' )
else :
return render_template ( ' errors/404.html ' ) , 404
2015-12-13 16:34:12 +07:00
2018-03-30 13:49:35 +07:00
2017-09-22 15:28:09 +01:00
@app.route ( ' /google/login ' )
def google_login ( ) :
if not app . config . get ( ' GOOGLE_OAUTH_ENABLE ' ) :
return abort ( 400 )
return google . authorize ( callback = url_for ( ' authorized ' , _external = True ) )
2016-08-05 16:20:41 +08:00
@app.route ( ' /github/login ' )
def github_login ( ) :
if not app . config . get ( ' GITHUB_OAUTH_ENABLE ' ) :
return abort ( 400 )
return github . authorize ( callback = url_for ( ' authorized ' , _external = True ) )
2017-10-31 19:21:22 +01:00
@app.route ( ' /saml/login ' )
def saml_login ( ) :
if not app . config . get ( ' SAML_ENABLED ' ) :
return abort ( 400 )
2017-10-31 22:38:26 +01:00
req = utils . prepare_flask_request ( request )
auth = utils . init_saml_auth ( req )
redirect_url = OneLogin_Saml2_Utils . get_self_url ( req ) + url_for ( ' saml_authorized ' )
return redirect ( auth . login ( return_to = redirect_url ) )
2017-10-31 19:21:22 +01:00
2017-10-31 22:38:26 +01:00
@app.route ( ' /saml/metadata ' )
2017-10-31 19:21:22 +01:00
def saml_metadata ( ) :
2017-10-31 22:38:26 +01:00
if not app . config . get ( ' SAML_ENABLED ' ) :
return abort ( 400 )
2017-10-31 19:21:22 +01:00
req = utils . prepare_flask_request ( request )
auth = utils . init_saml_auth ( req )
settings = auth . get_settings ( )
metadata = settings . get_sp_metadata ( )
errors = settings . validate_metadata ( metadata )
if len ( errors ) == 0 :
resp = make_response ( metadata , 200 )
resp . headers [ ' Content-Type ' ] = ' text/xml '
else :
resp = make_response ( errors . join ( ' , ' ) , 500 )
return resp
2017-10-31 22:38:26 +01:00
@app.route ( ' /saml/authorized ' , methods = [ ' GET ' , ' POST ' ] )
def saml_authorized ( ) :
errors = [ ]
if not app . config . get ( ' SAML_ENABLED ' ) :
return abort ( 400 )
req = utils . prepare_flask_request ( request )
auth = utils . init_saml_auth ( req )
auth . process_response ( )
2017-11-01 13:31:41 +01:00
errors = auth . get_errors ( )
2017-10-31 22:38:26 +01:00
if len ( errors ) == 0 :
session [ ' samlUserdata ' ] = auth . get_attributes ( )
session [ ' samlNameId ' ] = auth . get_nameid ( )
session [ ' samlSessionIndex ' ] = auth . get_session_index ( )
self_url = OneLogin_Saml2_Utils . get_self_url ( req )
self_url = self_url + req [ ' script_name ' ]
if ' RelayState ' in request . form and self_url != request . form [ ' RelayState ' ] :
return redirect ( auth . redirect_to ( request . form [ ' RelayState ' ] ) )
user = User . query . filter_by ( username = session [ ' samlNameId ' ] . lower ( ) ) . first ( )
if not user :
# create user
user = User ( username = session [ ' samlNameId ' ] ,
2017-11-06 23:36:11 +01:00
plain_text_password = None ,
2017-10-31 22:38:26 +01:00
email = session [ ' samlNameId ' ] )
user . create_local_user ( )
session [ ' user_id ' ] = user . id
if session [ ' samlUserdata ' ] . has_key ( " email " ) :
user . email = session [ ' samlUserdata ' ] [ " email " ] [ 0 ] . lower ( )
if session [ ' samlUserdata ' ] . has_key ( " givenname " ) :
user . firstname = session [ ' samlUserdata ' ] [ " givenname " ] [ 0 ]
if session [ ' samlUserdata ' ] . has_key ( " surname " ) :
user . lastname = session [ ' samlUserdata ' ] [ " surname " ] [ 0 ]
2017-11-06 23:36:11 +01:00
user . plain_text_password = None
2017-10-31 22:38:26 +01:00
user . update_profile ( )
2017-11-01 01:34:29 +01:00
session [ ' external_auth ' ] = True
2017-10-31 22:38:26 +01:00
login_user ( user , remember = False )
return redirect ( url_for ( ' index ' ) )
else :
2017-11-03 12:24:25 +01:00
return render_template ( ' errors/SAML.html ' , errors = errors )
2018-03-30 13:49:35 +07:00
2015-12-13 16:34:12 +07:00
@app.route ( ' /login ' , methods = [ ' GET ' , ' POST ' ] )
@login_manager.unauthorized_handler
def login ( ) :
2016-04-29 12:26:10 -06:00
LOGIN_TITLE = app . config [ ' LOGIN_TITLE ' ] if ' LOGIN_TITLE ' in app . config . keys ( ) else ' '
BASIC_ENABLED = app . config [ ' BASIC_ENABLED ' ]
SIGNUP_ENABLED = app . config [ ' SIGNUP_ENABLED ' ]
2018-04-02 13:38:53 +07:00
LDAP_ENABLED = app . config . get ( ' LDAP_ENABLED ' )
2016-08-05 16:20:41 +08:00
GITHUB_ENABLE = app . config . get ( ' GITHUB_OAUTH_ENABLE ' )
2017-09-22 15:28:09 +01:00
GOOGLE_ENABLE = app . config . get ( ' GOOGLE_OAUTH_ENABLE ' )
2017-10-31 19:21:22 +01:00
SAML_ENABLED = app . config . get ( ' SAML_ENABLED ' )
2015-12-13 16:34:12 +07:00
if g . user is not None and current_user . is_authenticated :
return redirect ( url_for ( ' dashboard ' ) )
2017-09-22 15:28:09 +01:00
if ' google_token ' in session :
user_data = google . get ( ' userinfo ' ) . data
first_name = user_data [ ' given_name ' ]
surname = user_data [ ' family_name ' ]
email = user_data [ ' email ' ]
user = User . query . filter_by ( username = email ) . first ( )
if not user :
# create user
user = User ( username = email ,
firstname = first_name ,
lastname = surname ,
2017-11-06 23:36:11 +01:00
plain_text_password = None ,
2017-09-22 15:28:09 +01:00
email = email )
2018-03-30 17:43:34 +07:00
result = user . create_local_user ( )
if not result [ ' status ' ] :
session . pop ( ' google_token ' , None )
return redirect ( url_for ( ' login ' ) )
2017-09-22 15:28:09 +01:00
session [ ' user_id ' ] = user . id
login_user ( user , remember = False )
2017-11-01 22:30:08 +01:00
session [ ' external_auth ' ] = True
2017-09-22 15:28:09 +01:00
return redirect ( url_for ( ' index ' ) )
2016-08-05 16:20:41 +08:00
if ' github_token ' in session :
me = github . get ( ' user ' )
user_info = me . data
user = User . query . filter_by ( username = user_info [ ' name ' ] ) . first ( )
if not user :
# create user
user = User ( username = user_info [ ' name ' ] ,
2017-11-06 23:36:11 +01:00
plain_text_password = None ,
2016-08-05 16:20:41 +08:00
email = user_info [ ' email ' ] )
2018-03-30 17:43:34 +07:00
result = user . create_local_user ( )
if not result [ ' status ' ] :
session . pop ( ' github_token ' , None )
return redirect ( url_for ( ' login ' ) )
2016-08-05 16:20:41 +08:00
session [ ' user_id ' ] = user . id
2017-11-01 01:34:29 +01:00
session [ ' external_auth ' ] = True
2016-08-05 16:20:41 +08:00
login_user ( user , remember = False )
return redirect ( url_for ( ' index ' ) )
2015-12-13 16:34:12 +07:00
if request . method == ' GET ' :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
# process login
username = request . form [ ' username ' ]
password = request . form [ ' password ' ]
2016-08-13 00:49:53 +08:00
otp_token = request . form . get ( ' otptoken ' )
auth_method = request . form . get ( ' auth_method ' , ' LOCAL ' )
2015-12-13 16:34:12 +07:00
# addition fields for registration case
2016-08-13 00:49:53 +08:00
firstname = request . form . get ( ' firstname ' )
lastname = request . form . get ( ' lastname ' )
email = request . form . get ( ' email ' )
rpassword = request . form . get ( ' rpassword ' )
2016-08-05 16:20:41 +08:00
2018-04-10 08:59:28 +07:00
if auth_method != ' LOCAL ' :
session [ ' external_auth ' ] = True
2015-12-13 16:34:12 +07:00
if None in [ firstname , lastname , email ] :
#login case
remember_me = False
if ' remember ' in request . form :
remember_me = True
user = User ( username = username , password = password , plain_text_password = password )
try :
auth = user . is_validate ( method = auth_method )
if auth == False :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , error = ' Invalid credentials ' ,
github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2018-03-30 13:49:35 +07:00
except Exception as e :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , error = e ,
github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2015-12-13 16:34:12 +07:00
2016-06-16 15:33:05 +07:00
# check if user enabled OPT authentication
if user . otp_secret :
2018-06-06 09:17:26 -02:00
if otp_token and otp_token . isdigit ( ) :
2016-06-16 15:33:05 +07:00
good_token = user . verify_totp ( otp_token )
if not good_token :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , error = ' Invalid credentials ' ,
github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2016-06-16 15:33:05 +07:00
else :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , error = ' Token required ' ,
github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2016-06-16 15:33:05 +07:00
2015-12-13 16:34:12 +07:00
login_user ( user , remember = remember_me )
return redirect ( request . args . get ( ' next ' ) or url_for ( ' index ' ) )
else :
2018-04-18 13:16:02 +07:00
if not username or not password or not email :
return render_template ( ' register.html ' , error = ' Please input required information ' )
2015-12-13 16:34:12 +07:00
# registration case
user = User ( username = username , plain_text_password = password , firstname = firstname , lastname = lastname , email = email )
2016-08-05 16:20:41 +08:00
2016-04-22 18:19:03 -06:00
# TODO: Move this into the JavaScript
# validate password and password confirmation
if password != rpassword :
2018-04-18 13:16:02 +07:00
error = " Password confirmation does not match "
2016-04-22 18:19:03 -06:00
return render_template ( ' register.html ' , error = error )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
try :
result = user . create_local_user ( )
if result == True :
2018-04-10 08:59:28 +07:00
return render_template ( ' login.html ' , username = username , password = password ,
github_enabled = GITHUB_ENABLE ,
google_enabled = GOOGLE_ENABLE ,
saml_enabled = SAML_ENABLED ,
ldap_enabled = LDAP_ENABLED ,
login_title = LOGIN_TITLE ,
basic_enabled = BASIC_ENABLED ,
signup_enabled = SIGNUP_ENABLED )
2015-12-13 16:34:12 +07:00
else :
2018-03-30 17:43:34 +07:00
return render_template ( ' register.html ' , error = result [ ' msg ' ] )
2018-03-30 13:49:35 +07:00
except Exception as e :
return render_template ( ' register.html ' , error = e )
2017-12-05 00:14:31 +01:00
def clear_session ( ) :
2016-08-13 00:49:53 +08:00
session . pop ( ' user_id ' , None )
session . pop ( ' github_token ' , None )
2017-09-22 15:28:09 +01:00
session . pop ( ' google_token ' , None )
2017-10-31 22:38:26 +01:00
session . clear ( )
2015-12-13 16:34:12 +07:00
logout_user ( )
2017-12-05 00:14:31 +01:00
@app.route ( ' /logout ' )
def logout ( ) :
2017-12-05 00:23:10 +01:00
if app . config . get ( ' SAML_ENABLED ' ) and ' samlSessionIndex ' in session and app . config . get ( ' SAML_LOGOUT ' ) :
2017-12-05 00:14:31 +01:00
req = utils . prepare_flask_request ( request )
auth = utils . init_saml_auth ( req )
if app . config . get ( ' SAML_LOGOUT_URL ' ) :
2017-12-05 03:48:18 +01:00
return redirect ( auth . logout ( name_id_format = " urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress " ,
2017-12-05 12:28:54 +01:00
return_to = app . config . get ( ' SAML_LOGOUT_URL ' ) ,
2017-12-05 03:48:18 +01:00
session_index = session [ ' samlSessionIndex ' ] , name_id = session [ ' samlNameId ' ] ) )
return redirect ( auth . logout ( name_id_format = " urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress " ,
session_index = session [ ' samlSessionIndex ' ] ,
name_id = session [ ' samlNameId ' ] ) )
2017-12-05 00:14:31 +01:00
clear_session ( )
2016-08-05 16:20:41 +08:00
return redirect ( url_for ( ' login ' ) )
2015-12-13 16:34:12 +07:00
2017-12-05 00:14:31 +01:00
@app.route ( ' /saml/sls ' )
def saml_logout ( ) :
req = utils . prepare_flask_request ( request )
auth = utils . init_saml_auth ( req )
2017-12-05 03:48:18 +01:00
url = auth . process_slo ( )
2017-12-05 00:14:31 +01:00
errors = auth . get_errors ( )
if len ( errors ) == 0 :
2017-12-05 03:48:18 +01:00
clear_session ( )
2017-12-05 00:14:31 +01:00
if url is not None :
return redirect ( url )
2018-01-20 17:17:02 +01:00
elif app . config . get ( ' SAML_LOGOUT_URL ' ) is not None :
2017-12-05 12:59:08 +01:00
return redirect ( app . config . get ( ' SAML_LOGOUT_URL ' ) )
2018-01-20 17:17:02 +01:00
else :
return redirect ( url_for ( ' login ' ) )
2017-12-05 00:14:31 +01:00
else :
return render_template ( ' errors/SAML.html ' , errors = errors )
2015-12-13 16:34:12 +07:00
@app.route ( ' /dashboard ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
def dashboard ( ) :
2018-06-08 11:46:17 +02:00
if not app . config . get ( ' BG_DOMAIN_UPDATES ' ) :
logging . debug ( ' Update domains in foreground ' )
d = Domain ( ) . update ( )
else :
logging . debug ( ' Update domains in background ' )
2016-04-27 17:19:25 -06:00
# stats for dashboard
2016-04-29 10:29:08 -06:00
domain_count = Domain . query . count ( )
2016-04-27 17:19:25 -06:00
users = User . query . all ( )
history_number = History . query . count ( )
2016-06-30 22:04:59 +02:00
history = History . query . order_by ( History . created_on . desc ( ) ) . limit ( 4 )
2016-04-27 17:19:25 -06:00
server = Server ( server_id = ' localhost ' )
statistics = server . get_statistic ( )
if statistics :
2018-03-30 13:49:35 +07:00
uptime = list ( [ uptime for uptime in statistics if uptime [ ' name ' ] == ' uptime ' ] ) [ 0 ] [ ' value ' ]
2016-04-27 17:19:25 -06:00
else :
uptime = 0
2018-06-06 08:42:57 -02:00
2018-06-08 13:22:03 +02:00
return render_template ( ' dashboard.html ' , domain_count = domain_count , users = users , history_number = history_number , uptime = uptime , histories = history , dnssec_adm_only = app . config [ ' DNSSEC_ADMINS_ONLY ' ] , pdns_version = app . config [ ' PDNS_VERSION ' ] , show_bg_domain_button = app . config [ ' BG_DOMAIN_UPDATES ' ] )
2017-06-30 18:18:06 +02:00
@app.route ( ' /dashboard-domains ' , methods = [ ' GET ' ] )
@login_required
def dashboard_domains ( ) :
if current_user . role . name == ' Administrator ' :
domains = Domain . query
else :
2017-09-15 15:14:04 +02:00
domains = User ( id = current_user . id ) . get_domain_query ( )
2017-06-30 18:18:06 +02:00
template = app . jinja_env . get_template ( " dashboard_domain.html " )
render = template . make_module ( vars = { " current_user " : current_user } )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
columns = [ Domain . name , Domain . dnssec , Domain . type , Domain . serial , Domain . master , Domain . account ]
2017-06-30 18:18:06 +02:00
# History.created_on.desc()
order_by = [ ]
for i in range ( len ( columns ) ) :
2018-04-01 07:57:41 +07:00
column_index = request . args . get ( " order[ {0} ][column] " . format ( i ) )
sort_direction = request . args . get ( " order[ {0} ][dir] " . format ( i ) )
2017-06-30 18:18:06 +02:00
if column_index is None :
break
if sort_direction != " asc " and sort_direction != " desc " :
sort_direction = " asc "
column = columns [ int ( column_index ) ]
order_by . append ( getattr ( column , sort_direction ) ( ) )
if order_by :
domains = domains . order_by ( * order_by )
total_count = domains . count ( )
search = request . args . get ( " search[value] " )
if search :
start = " " if search . startswith ( " ^ " ) else " % "
end = " " if search . endswith ( " $ " ) else " % "
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
if current_user . role . name == ' Administrator ' :
domains = domains . outerjoin ( Account ) . filter ( Domain . name . ilike ( start + search . strip ( " ^$ " ) + end ) |
Account . name . ilike ( start + search . strip ( " ^$ " ) + end ) |
Account . description . ilike ( start + search . strip ( " ^$ " ) + end ) )
else :
domains = domains . filter ( Domain . name . ilike ( start + search . strip ( " ^$ " ) + end ) )
2017-06-30 18:18:06 +02:00
filtered_count = domains . count ( )
start = int ( request . args . get ( " start " , 0 ) )
length = min ( int ( request . args . get ( " length " , 0 ) ) , 100 )
2018-03-30 15:40:43 +07:00
if length != - 1 :
domains = domains [ start : start + length ]
2017-06-30 18:18:06 +02:00
data = [ ]
for domain in domains :
data . append ( [
render . name ( domain ) ,
render . dnssec ( domain ) ,
render . type ( domain ) ,
render . serial ( domain ) ,
render . master ( domain ) ,
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
render . account ( domain ) ,
2017-06-30 18:18:06 +02:00
render . actions ( domain ) ,
] )
response_data = {
" draw " : int ( request . args . get ( " draw " , 0 ) ) ,
" recordsTotal " : total_count ,
" recordsFiltered " : filtered_count ,
" data " : data ,
}
return jsonify ( response_data )
2015-12-13 16:34:12 +07:00
2018-06-08 13:21:17 +02:00
@app.route ( ' /dashboard-domains-updater ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
def dashboard_domains_updater ( ) :
logging . debug ( ' Update domains in background ' )
d = Domain ( ) . update ( )
response_data = {
" result " : d ,
}
return jsonify ( response_data )
2015-12-13 16:34:12 +07:00
2016-08-02 10:59:31 +02:00
@app.route ( ' /domain/<path:domain_name> ' , methods = [ ' GET ' , ' POST ' ] )
2015-12-13 16:34:12 +07:00
@login_required
2018-03-31 07:32:46 +07:00
@can_access_domain
2015-12-13 16:34:12 +07:00
def domain ( domain_name ) :
r = Record ( )
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
2017-10-20 12:53:03 +02:00
if not domain :
return redirect ( url_for ( ' error ' , code = 404 ) )
2015-12-13 16:34:12 +07:00
2017-10-20 12:53:03 +02:00
# query domain info from PowerDNS API
zone_info = r . get_record_data ( domain . name )
if zone_info :
jrecords = zone_info [ ' records ' ]
else :
# can not get any record, API server might be down
return redirect ( url_for ( ' error ' , code = 500 ) )
2018-06-22 14:55:23 -02:00
quick_edit = strtobool ( Setting ( ) . get ( ' allow_quick_edit ' ) )
2017-10-20 12:53:03 +02:00
records = [ ]
#TODO: This should be done in the "model" instead of "view"
if NEW_SCHEMA :
for jr in jrecords :
if jr [ ' type ' ] in app . config [ ' RECORDS_ALLOW_EDIT ' ] :
for subrecord in jr [ ' records ' ] :
record = Record ( name = jr [ ' name ' ] , type = jr [ ' type ' ] , status = ' Disabled ' if subrecord [ ' disabled ' ] else ' Active ' , ttl = jr [ ' ttl ' ] , data = subrecord [ ' content ' ] )
2016-06-08 11:00:55 +07:00
records . append ( record )
2016-11-21 15:52:07 +01:00
if not re . search ( ' ip6 \ .arpa|in-addr \ .arpa$ ' , domain_name ) :
editable_records = app . config [ ' RECORDS_ALLOW_EDIT ' ]
else :
2018-04-06 13:22:09 +07:00
editable_records = app . config [ ' REVERSE_RECORDS_ALLOW_EDIT ' ]
2018-06-22 14:55:23 -02:00
return render_template ( ' domain.html ' , domain = domain , records = records , editable_records = editable_records , quick_edit = quick_edit )
2015-12-13 16:34:12 +07:00
else :
2017-10-20 12:53:03 +02:00
for jr in jrecords :
if jr [ ' type ' ] in app . config [ ' RECORDS_ALLOW_EDIT ' ] :
record = Record ( name = jr [ ' name ' ] , type = jr [ ' type ' ] , status = ' Disabled ' if jr [ ' disabled ' ] else ' Active ' , ttl = jr [ ' ttl ' ] , data = jr [ ' content ' ] )
records . append ( record )
if not re . search ( ' ip6 \ .arpa|in-addr \ .arpa$ ' , domain_name ) :
2018-02-16 21:02:16 +02:00
editable_records = app . config [ ' FORWARD_RECORDS_ALLOW_EDIT ' ]
2017-10-20 12:53:03 +02:00
else :
2018-02-16 21:02:16 +02:00
editable_records = app . config [ ' REVERSE_RECORDS_ALLOW_EDIT ' ]
2018-06-22 14:55:23 -02:00
return render_template ( ' domain.html ' , domain = domain , records = records , editable_records = editable_records , quick_edit = quick_edit , pdns_version = app . config [ ' PDNS_VERSION ' ] )
2015-12-13 16:34:12 +07:00
@app.route ( ' /admin/domain/add ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def domain_add ( ) :
2018-01-22 18:22:19 +03:00
templates = DomainTemplate . query . all ( )
2015-12-13 16:34:12 +07:00
if request . method == ' POST ' :
try :
domain_name = request . form . getlist ( ' domain_name ' ) [ 0 ]
domain_type = request . form . getlist ( ' radio_type ' ) [ 0 ]
2018-01-22 18:22:19 +03:00
domain_template = request . form . getlist ( ' domain_template ' ) [ 0 ]
2016-03-05 17:04:12 +07:00
soa_edit_api = request . form . getlist ( ' radio_type_soa_edit_api ' ) [ 0 ]
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account_id = request . form . getlist ( ' accountid ' ) [ 0 ]
2016-01-15 11:58:53 +07:00
if ' ' in domain_name or not domain_name or not domain_type :
2016-05-15 16:01:57 -06:00
return render_template ( ' errors/400.html ' , msg = " Please correct your input " ) , 400
2016-01-15 11:58:53 +07:00
2015-12-13 16:34:12 +07:00
if domain_type == ' slave ' :
if request . form . getlist ( ' domain_master_address ' ) :
domain_master_string = request . form . getlist ( ' domain_master_address ' ) [ 0 ]
domain_master_string = domain_master_string . replace ( ' ' , ' ' )
domain_master_ips = domain_master_string . split ( ' , ' )
else :
domain_master_ips = [ ]
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account_name = Account ( ) . get_name_by_id ( account_id )
2015-12-13 16:34:12 +07:00
d = Domain ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
result = d . add ( domain_name = domain_name , domain_type = domain_type , soa_edit_api = soa_edit_api , domain_master_ips = domain_master_ips , account_name = account_name )
2015-12-13 16:34:12 +07:00
if result [ ' status ' ] == ' ok ' :
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
history = History ( msg = ' Add domain {0} ' . format ( domain_name ) , detail = str ( { ' domain_type ' : domain_type , ' domain_master_ips ' : domain_master_ips , ' account_id ' : account_id } ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
2018-01-22 18:22:19 +03:00
if domain_template != ' 0 ' :
template = DomainTemplate . query . filter ( DomainTemplate . id == domain_template ) . first ( )
template_records = DomainTemplateRecord . query . filter ( DomainTemplateRecord . template_id == domain_template ) . all ( )
record_data = [ ]
for template_record in template_records :
record_row = { ' record_data ' : template_record . data , ' record_name ' : template_record . name , ' record_status ' : template_record . status , ' record_ttl ' : template_record . ttl , ' record_type ' : template_record . type }
record_data . append ( record_row )
r = Record ( )
result = r . apply ( domain_name , record_data )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Applying template {0} to {1} , created records successfully. ' . format ( template . name , domain_name ) , detail = str ( result ) , created_by = current_user . username )
2018-01-22 18:22:19 +03:00
history . add ( )
else :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Applying template {0} to {1} , FAILED to created records. ' . format ( template . name , domain_name ) , detail = str ( result ) , created_by = current_user . username )
2018-01-22 18:22:19 +03:00
history . add ( )
2015-12-13 16:34:12 +07:00
return redirect ( url_for ( ' dashboard ' ) )
else :
2016-05-15 16:01:57 -06:00
return render_template ( ' errors/400.html ' , msg = result [ ' msg ' ] ) , 400
2015-12-13 16:34:12 +07:00
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2015-12-13 16:34:12 +07:00
return redirect ( url_for ( ' error ' , code = 500 ) )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
else :
accounts = Account . query . all ( )
return render_template ( ' domain_add.html ' , templates = templates , accounts = accounts )
2015-12-13 16:34:12 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /admin/domain/<path:domain_name>/delete ' , methods = [ ' GET ' ] )
2015-12-13 16:34:12 +07:00
@login_required
@admin_role_required
def domain_delete ( domain_name ) :
d = Domain ( )
result = d . delete ( domain_name )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
if result [ ' status ' ] == ' error ' :
return redirect ( url_for ( ' error ' , code = 500 ) )
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Delete domain {0} ' . format ( domain_name ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return redirect ( url_for ( ' dashboard ' ) )
2018-03-31 06:52:14 +07:00
@app.route ( ' /admin/domain/<path:domain_name>/manage ' , methods = [ ' GET ' , ' POST ' ] )
2015-12-13 16:34:12 +07:00
@login_required
@admin_role_required
def domain_management ( domain_name ) :
if request . method == ' GET ' :
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
if not domain :
return redirect ( url_for ( ' error ' , code = 404 ) )
users = User . query . all ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
accounts = Account . query . all ( )
2015-12-13 16:34:12 +07:00
# get list of user ids to initilize selection data
d = Domain ( name = domain_name )
domain_user_ids = d . get_user ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account = d . get_account ( )
2015-12-13 16:34:12 +07:00
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
return render_template ( ' domain_management.html ' , domain = domain , users = users , domain_user_ids = domain_user_ids , accounts = accounts , domain_account = account )
2015-12-13 16:34:12 +07:00
if request . method == ' POST ' :
# username in right column
new_user_list = request . form . getlist ( ' domain_multi_user[] ' )
# get list of user ids to compare
d = Domain ( name = domain_name )
domain_user_ids = d . get_user ( )
2016-08-05 16:20:41 +08:00
# grant/revoke user privielges
2015-12-13 16:34:12 +07:00
d . grant_privielges ( new_user_list )
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Change domain {0} access control ' . format ( domain_name ) , detail = str ( { ' user_has_access ' : new_user_list } ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return redirect ( url_for ( ' domain_management ' , domain_name = domain_name ) )
2016-03-05 17:04:12 +07:00
2018-04-02 13:38:53 +07:00
@app.route ( ' /admin/domain/<path:domain_name>/change_soa_setting ' , methods = [ ' POST ' ] )
2018-03-28 01:41:33 +02:00
@login_required
@admin_role_required
def domain_change_soa_edit_api ( domain_name ) :
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
if not domain :
return redirect ( url_for ( ' error ' , code = 404 ) )
new_setting = request . form . get ( ' soa_edit_api ' )
if new_setting == None :
return redirect ( url_for ( ' error ' , code = 500 ) )
if new_setting == ' 0 ' :
return redirect ( url_for ( ' domain_management ' , domain_name = domain_name ) )
2018-04-02 13:38:53 +07:00
2018-03-28 01:41:33 +02:00
d = Domain ( )
status = d . update_soa_setting ( domain_name = domain_name , soa_edit_api = new_setting )
if status [ ' status ' ] != None :
users = User . query . all ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
accounts = Account . query . all ( )
2018-03-28 01:41:33 +02:00
d = Domain ( name = domain_name )
domain_user_ids = d . get_user ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account = d . get_account ( )
return render_template ( ' domain_management.html ' , domain = domain , users = users , domain_user_ids = domain_user_ids , accounts = accounts , domain_account = account )
else :
return redirect ( url_for ( ' error ' , code = 500 ) )
@app.route ( ' /admin/domain/<path:domain_name>/change_account ' , methods = [ ' POST ' ] )
@login_required
@admin_role_required
def domain_change_account ( domain_name ) :
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
if not domain :
return redirect ( url_for ( ' error ' , code = 404 ) )
account_id = request . form . get ( ' accountid ' )
status = domain . assoc_account ( account_id )
if status [ ' status ' ] :
users = User . query . all ( )
accounts = Account . query . all ( )
2018-03-28 01:41:33 +02:00
d = Domain ( name = domain_name )
domain_user_ids = d . get_user ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account = d . get_account ( )
return render_template ( ' domain_management.html ' , domain = domain , users = users , domain_user_ids = domain_user_ids , accounts = accounts , domain_account = account )
2018-03-28 01:41:33 +02:00
else :
return redirect ( url_for ( ' error ' , code = 500 ) )
2016-03-05 17:04:12 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /domain/<path:domain_name>/apply ' , methods = [ ' POST ' ] , strict_slashes = False )
2015-12-13 16:34:12 +07:00
@login_required
2018-03-31 07:32:46 +07:00
@can_access_domain
2015-12-13 16:34:12 +07:00
def record_apply ( domain_name ) :
"""
example jdata : { u ' record_ttl ' : u ' 1800 ' , u ' record_type ' : u ' CNAME ' , u ' record_name ' : u ' test4 ' , u ' record_status ' : u ' Active ' , u ' record_data ' : u ' duykhanh.me ' }
"""
#TODO: filter removed records / name modified records.
2018-04-12 11:18:44 +07:00
2015-12-13 16:34:12 +07:00
try :
2018-03-30 13:49:35 +07:00
jdata = request . json
2015-12-13 16:34:12 +07:00
2018-04-12 11:18:44 +07:00
submitted_serial = jdata [ ' serial ' ]
submitted_record = jdata [ ' record ' ]
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
logging . debug ( ' Your submitted serial: {0} ' . format ( submitted_serial ) )
logging . debug ( ' Current domain serial: {0} ' . format ( domain . serial ) )
if domain :
if int ( submitted_serial ) != domain . serial :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' The zone has been changed by another session or user. Please refresh this web page to load updated records. ' } ) , 500 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Domain name {0} does not exist ' . format ( domain_name ) } ) , 404 )
2015-12-13 16:34:12 +07:00
r = Record ( )
2018-04-12 11:18:44 +07:00
result = r . apply ( domain_name , submitted_record )
2015-12-13 16:34:12 +07:00
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Apply record changes to domain {0} ' . format ( domain_name ) , detail = str ( jdata ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return make_response ( jsonify ( result ) , 200 )
else :
return make_response ( jsonify ( result ) , 400 )
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2015-12-13 16:34:12 +07:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Error when applying new changes ' } ) , 500 )
2018-03-31 06:52:14 +07:00
@app.route ( ' /domain/<path:domain_name>/update ' , methods = [ ' POST ' ] , strict_slashes = False )
2015-12-13 16:34:12 +07:00
@login_required
2018-03-31 07:32:46 +07:00
@can_access_domain
2015-12-13 16:34:12 +07:00
def record_update ( domain_name ) :
"""
This route is used for domain work as Slave Zone only
Pulling the records update from its Master
"""
try :
2018-03-30 13:49:35 +07:00
jdata = request . json
2015-12-13 16:34:12 +07:00
domain_name = jdata [ ' domain ' ]
d = Domain ( )
result = d . update_from_master ( domain_name )
if result [ ' status ' ] == ' ok ' :
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : result [ ' msg ' ] } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : result [ ' msg ' ] } ) , 500 )
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2015-12-13 16:34:12 +07:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Error when applying new changes ' } ) , 500 )
2018-03-31 06:52:14 +07:00
@app.route ( ' /domain/<path:domain_name>/record/<path:record_name>/type/<path:record_type>/delete ' , methods = [ ' GET ' ] )
2015-12-13 16:34:12 +07:00
@login_required
@admin_role_required
def record_delete ( domain_name , record_name , record_type ) :
try :
r = Record ( name = record_name , type = record_type )
result = r . delete ( domain = domain_name )
if result [ ' status ' ] == ' error ' :
2018-03-30 13:49:35 +07:00
print ( result [ ' msg ' ] )
2015-12-13 16:34:12 +07:00
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2016-08-05 16:20:41 +08:00
return redirect ( url_for ( ' error ' , code = 500 ) ) , 500
2015-12-13 16:34:12 +07:00
return redirect ( url_for ( ' domain ' , domain_name = domain_name ) )
2018-04-12 11:18:44 +07:00
@app.route ( ' /domain/<path:domain_name>/info ' , methods = [ ' GET ' ] )
@login_required
@can_access_domain
def domain_info ( domain_name ) :
domain = Domain ( )
domain_info = domain . get_domain_info ( domain_name )
return make_response ( jsonify ( domain_info ) , 200 )
2018-03-31 06:52:14 +07:00
@app.route ( ' /domain/<path:domain_name>/dnssec ' , methods = [ ' GET ' ] )
2016-03-24 20:01:08 +07:00
@login_required
2018-04-02 13:38:53 +07:00
@can_access_domain
2016-03-24 20:01:08 +07:00
def domain_dnssec ( domain_name ) :
domain = Domain ( )
dnssec = domain . get_domain_dnssec ( domain_name )
return make_response ( jsonify ( dnssec ) , 200 )
2018-03-30 13:49:35 +07:00
2018-04-02 13:38:53 +07:00
@app.route ( ' /domain/<path:domain_name>/dnssec/enable ' , methods = [ ' GET ' ] )
2018-03-01 08:27:10 +01:00
@login_required
2018-04-02 13:38:53 +07:00
@can_access_domain
2018-06-07 09:28:14 +07:00
@can_configure_dnssec
2018-03-01 08:27:10 +01:00
def domain_dnssec_enable ( domain_name ) :
domain = Domain ( )
dnssec = domain . enable_domain_dnssec ( domain_name )
return make_response ( jsonify ( dnssec ) , 200 )
2018-04-02 13:38:53 +07:00
@app.route ( ' /domain/<path:domain_name>/dnssec/disable ' , methods = [ ' GET ' ] )
2018-03-01 08:27:10 +01:00
@login_required
2018-04-02 13:38:53 +07:00
@can_access_domain
2018-06-07 09:28:14 +07:00
@can_configure_dnssec
2018-03-01 08:27:10 +01:00
def domain_dnssec_disable ( domain_name ) :
domain = Domain ( )
dnssec = domain . get_domain_dnssec ( domain_name )
2018-03-05 15:06:40 +01:00
for key in dnssec [ ' dnssec ' ] :
response = domain . delete_dnssec_key ( domain_name , key [ ' id ' ] ) ;
2018-03-05 15:11:42 +01:00
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' DNSSEC removed. ' } ) )
2018-03-05 15:06:40 +01:00
2018-03-01 08:27:10 +01:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /domain/<path:domain_name>/managesetting ' , methods = [ ' GET ' , ' POST ' ] )
2016-07-02 11:24:13 -06:00
@login_required
@admin_role_required
def admin_setdomainsetting ( domain_name ) :
if request . method == ' POST ' :
#
# post data should in format
# {'action': 'set_setting', 'setting': 'default_action, 'value': 'True'}
#
try :
2018-03-30 13:49:35 +07:00
jdata = request . json
2016-07-02 11:24:13 -06:00
data = jdata [ ' data ' ]
2018-03-30 13:49:35 +07:00
2016-07-02 11:24:13 -06:00
if jdata [ ' action ' ] == ' set_setting ' :
new_setting = data [ ' setting ' ]
2016-07-02 22:30:16 -06:00
new_value = str ( data [ ' value ' ] )
2016-07-02 11:24:13 -06:00
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
setting = DomainSetting . query . filter ( DomainSetting . domain == domain ) . filter ( DomainSetting . setting == new_setting ) . first ( )
2016-08-05 16:20:41 +08:00
2016-07-02 11:24:13 -06:00
if setting :
if setting . set ( new_value ) :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Setting {0} changed value to {1} for {2} ' . format ( new_setting , new_value , domain . name ) , created_by = current_user . username )
2016-07-02 11:24:13 -06:00
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Setting updated. ' } ) )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Unable to set value of setting. ' } ) )
else :
if domain . add_setting ( new_setting , new_value ) :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' New setting {0} with value {1} for {2} has been created ' . format ( new_setting , new_value , domain . name ) , created_by = current_user . username )
2016-07-02 11:24:13 -06:00
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' New setting created and updated. ' } ) )
else :
2016-08-05 16:20:41 +08:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Unable to create new setting. ' } ) )
2016-07-02 11:24:13 -06:00
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Action not supported. ' } ) , 400 )
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2016-07-02 11:24:13 -06:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' There is something wrong, please contact Administrator. ' } ) , 400 )
2016-03-24 20:01:08 +07:00
2018-01-22 18:22:19 +03:00
@app.route ( ' /templates ' , methods = [ ' GET ' , ' POST ' ] )
@app.route ( ' /templates/list ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def templates ( ) :
templates = DomainTemplate . query . all ( )
return render_template ( ' template.html ' , templates = templates )
@app.route ( ' /template/create ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def create_template ( ) :
if request . method == ' GET ' :
return render_template ( ' template_add.html ' )
if request . method == ' POST ' :
try :
name = request . form . getlist ( ' name ' ) [ 0 ]
description = request . form . getlist ( ' description ' ) [ 0 ]
if ' ' in name or not name or not type :
flash ( " Please correct your input " , ' error ' )
return redirect ( url_for ( ' create_template ' ) )
if DomainTemplate . query . filter ( DomainTemplate . name == name ) . first ( ) :
2018-04-01 07:57:41 +07:00
flash ( " A template with the name {0} already exists! " . format ( name ) , ' error ' )
2018-01-22 18:22:19 +03:00
return redirect ( url_for ( ' create_template ' ) )
t = DomainTemplate ( name = name , description = description )
result = t . create ( )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Add domain template {0} ' . format ( name ) , detail = str ( { ' name ' : name , ' description ' : description } ) , created_by = current_user . username )
2018-01-22 18:22:19 +03:00
history . add ( )
return redirect ( url_for ( ' templates ' ) )
else :
flash ( result [ ' msg ' ] , ' error ' )
return redirect ( url_for ( ' create_template ' ) )
2018-03-31 08:21:02 +07:00
except :
logging . error ( traceback . print_exc ( ) )
2018-01-22 18:22:19 +03:00
return redirect ( url_for ( ' error ' , code = 500 ) )
return redirect ( url_for ( ' templates ' ) )
2018-01-23 18:23:21 +03:00
@app.route ( ' /template/createfromzone ' , methods = [ ' POST ' ] )
@login_required
@admin_role_required
def create_template_from_zone ( ) :
try :
2018-03-31 08:21:02 +07:00
jdata = request . json
2018-01-23 18:23:21 +03:00
name = jdata [ ' name ' ]
description = jdata [ ' description ' ]
domain_name = jdata [ ' domain ' ]
if ' ' in name or not name or not type :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Please correct template name ' } ) , 500 )
if DomainTemplate . query . filter ( DomainTemplate . name == name ) . first ( ) :
2018-04-01 07:57:41 +07:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' A template with the name {0} already exists! ' . format ( name ) } ) , 500 )
2018-01-23 18:23:21 +03:00
t = DomainTemplate ( name = name , description = description )
result = t . create ( )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Add domain template {0} ' . format ( name ) , detail = str ( { ' name ' : name , ' description ' : description } ) , created_by = current_user . username )
2018-01-23 18:23:21 +03:00
history . add ( )
records = [ ]
r = Record ( )
domain = Domain . query . filter ( Domain . name == domain_name ) . first ( )
if domain :
# query domain info from PowerDNS API
zone_info = r . get_record_data ( domain . name )
if zone_info :
jrecords = zone_info [ ' records ' ]
if NEW_SCHEMA :
for jr in jrecords :
if jr [ ' type ' ] in app . config [ ' RECORDS_ALLOW_EDIT ' ] :
2018-04-12 02:20:49 +02:00
name = ' @ ' if jr [ ' name ' ] == domain_name else re . sub ( ' \ . {} $ ' . format ( domain_name ) , ' ' , jr [ ' name ' ] )
2018-01-23 18:23:21 +03:00
for subrecord in jr [ ' records ' ] :
record = DomainTemplateRecord ( name = name , type = jr [ ' type ' ] , status = True if subrecord [ ' disabled ' ] else False , ttl = jr [ ' ttl ' ] , data = subrecord [ ' content ' ] )
records . append ( record )
else :
for jr in jrecords :
if jr [ ' type ' ] in app . config [ ' RECORDS_ALLOW_EDIT ' ] :
2018-04-12 02:20:49 +02:00
name = ' @ ' if jr [ ' name ' ] == domain_name else re . sub ( ' \ . {} $ ' . format ( domain_name ) , ' ' , jr [ ' name ' ] )
2018-01-23 18:23:21 +03:00
record = DomainTemplateRecord ( name = name , type = jr [ ' type ' ] , status = True if jr [ ' disabled ' ] else False , ttl = jr [ ' ttl ' ] , data = jr [ ' content ' ] )
records . append ( record )
2018-04-11 18:09:38 +02:00
2018-01-23 18:23:21 +03:00
result_records = t . replace_records ( records )
if result_records [ ' status ' ] == ' ok ' :
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : result [ ' msg ' ] } ) , 200 )
else :
result = t . delete_template ( )
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : result_records [ ' msg ' ] } ) , 500 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : result [ ' msg ' ] } ) , 500 )
2018-03-31 08:21:02 +07:00
except :
logging . error ( traceback . print_exc ( ) )
2018-01-23 18:23:21 +03:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Error when applying new changes ' } ) , 500 )
2018-04-02 13:38:53 +07:00
@app.route ( ' /template/<path:template>/edit ' , methods = [ ' GET ' ] )
2018-01-22 18:22:19 +03:00
@login_required
@admin_role_required
def edit_template ( template ) :
try :
t = DomainTemplate . query . filter ( DomainTemplate . name == template ) . first ( )
if t is not None :
records = [ ]
for jr in t . records :
if jr . type in app . config [ ' RECORDS_ALLOW_EDIT ' ] :
record = DomainTemplateRecord ( name = jr . name , type = jr . type , status = ' Disabled ' if jr . status else ' Active ' , ttl = jr . ttl , data = jr . data )
records . append ( record )
return render_template ( ' template_edit.html ' , template = t . name , records = records , editable_records = app . config [ ' RECORDS_ALLOW_EDIT ' ] )
2018-03-31 08:21:02 +07:00
except :
logging . error ( traceback . print_exc ( ) )
2018-01-22 18:22:19 +03:00
return redirect ( url_for ( ' error ' , code = 500 ) )
return redirect ( url_for ( ' templates ' ) )
2018-04-02 13:38:53 +07:00
@app.route ( ' /template/<path:template>/apply ' , methods = [ ' POST ' ] , strict_slashes = False )
2018-01-22 18:22:19 +03:00
@login_required
def apply_records ( template ) :
try :
2018-03-31 08:21:02 +07:00
jdata = request . json
2018-01-22 18:22:19 +03:00
records = [ ]
for j in jdata :
name = ' @ ' if j [ ' record_name ' ] in [ ' @ ' , ' ' ] else j [ ' record_name ' ]
type = j [ ' record_type ' ]
data = j [ ' record_data ' ]
disabled = True if j [ ' record_status ' ] == ' Disabled ' else False
ttl = int ( j [ ' record_ttl ' ] ) if j [ ' record_ttl ' ] else 3600
dtr = DomainTemplateRecord ( name = name , type = type , data = data , status = disabled , ttl = ttl )
records . append ( dtr )
t = DomainTemplate . query . filter ( DomainTemplate . name == template ) . first ( )
result = t . replace_records ( records )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Apply domain template record changes to domain template {0} ' . format ( template ) , detail = str ( jdata ) , created_by = current_user . username )
2018-01-22 18:22:19 +03:00
history . add ( )
return make_response ( jsonify ( result ) , 200 )
else :
return make_response ( jsonify ( result ) , 400 )
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2018-01-22 18:22:19 +03:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Error when applying new changes ' } ) , 500 )
2018-04-02 13:38:53 +07:00
@app.route ( ' /template/<path:template>/delete ' , methods = [ ' GET ' ] )
2018-01-22 18:22:19 +03:00
@login_required
@admin_role_required
def delete_template ( template ) :
try :
t = DomainTemplate . query . filter ( DomainTemplate . name == template ) . first ( )
if t is not None :
result = t . delete_template ( )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Deleted domain template {0} ' . format ( template ) , detail = str ( { ' name ' : template } ) , created_by = current_user . username )
2018-01-22 18:22:19 +03:00
history . add ( )
return redirect ( url_for ( ' templates ' ) )
else :
flash ( result [ ' msg ' ] , ' error ' )
return redirect ( url_for ( ' templates ' ) )
2018-03-31 08:21:02 +07:00
except :
logging . error ( traceback . print_exc ( ) )
2018-01-22 18:22:19 +03:00
return redirect ( url_for ( ' error ' , code = 500 ) )
return redirect ( url_for ( ' templates ' ) )
2015-12-13 16:34:12 +07:00
@app.route ( ' /admin ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin ( ) :
domains = Domain . query . all ( )
users = User . query . all ( )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
server = Server ( server_id = ' localhost ' )
configs = server . get_config ( )
statistics = server . get_statistic ( )
history_number = History . query . count ( )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
if statistics :
2018-03-30 13:49:35 +07:00
uptime = list ( [ uptime for uptime in statistics if uptime [ ' name ' ] == ' uptime ' ] ) [ 0 ] [ ' value ' ]
2015-12-13 16:34:12 +07:00
else :
uptime = 0
return render_template ( ' admin.html ' , domains = domains , users = users , configs = configs , statistics = statistics , uptime = uptime , history_number = history_number )
2018-03-30 13:49:35 +07:00
2016-05-15 14:29:15 -06:00
@app.route ( ' /admin/user/create ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin_createuser ( ) :
if request . method == ' GET ' :
return render_template ( ' admin_createuser.html ' )
2016-08-05 16:20:41 +08:00
2016-05-15 14:29:15 -06:00
if request . method == ' POST ' :
fdata = request . form
2016-08-05 16:20:41 +08:00
2016-05-15 14:29:15 -06:00
user = User ( username = fdata [ ' username ' ] , plain_text_password = fdata [ ' password ' ] , firstname = fdata [ ' firstname ' ] , lastname = fdata [ ' lastname ' ] , email = fdata [ ' email ' ] )
2016-08-05 16:20:41 +08:00
2016-05-15 14:29:15 -06:00
if fdata [ ' password ' ] == " " :
return render_template ( ' admin_createuser.html ' , user = user , blank_password = True )
2016-08-05 16:20:41 +08:00
2016-05-15 14:29:15 -06:00
result = user . create_local_user ( ) ;
2018-03-30 17:43:34 +07:00
if result [ ' status ' ] :
return redirect ( url_for ( ' admin_manageuser ' ) )
2016-08-05 16:20:41 +08:00
2018-03-30 17:43:34 +07:00
return render_template ( ' admin_createuser.html ' , user = user , error = result [ ' msg ' ] )
2015-12-13 16:34:12 +07:00
2018-03-30 13:49:35 +07:00
2015-12-13 16:34:12 +07:00
@app.route ( ' /admin/manageuser ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin_manageuser ( ) :
if request . method == ' GET ' :
2016-05-15 14:29:15 -06:00
users = User . query . order_by ( User . username ) . all ( )
2015-12-13 16:34:12 +07:00
return render_template ( ' admin_manageuser.html ' , users = users )
if request . method == ' POST ' :
#
# post data should in format
# {'action': 'delete_user', 'data': 'username'}
#
try :
2018-03-30 13:49:35 +07:00
jdata = request . json
2015-12-13 16:34:12 +07:00
data = jdata [ ' data ' ]
if jdata [ ' action ' ] == ' delete_user ' :
user = User ( username = data )
2018-06-07 15:26:54 -02:00
if user . username == current_user . username :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' You cannot delete yourself. ' } ) , 400 )
2015-12-13 16:34:12 +07:00
result = user . delete ( )
if result :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Delete username {0} ' . format ( data ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' User has been removed. ' } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Cannot remove user. ' } ) , 500 )
2016-08-05 16:20:41 +08:00
2015-12-13 16:34:12 +07:00
elif jdata [ ' action ' ] == ' revoke_user_privielges ' :
user = User ( username = data )
result = user . revoke_privilege ( )
if result :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Revoke {0} user privielges ' . format ( data ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Revoked user privielges. ' } ) , 200 )
else :
2016-08-05 16:20:41 +08:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Cannot revoke user privilege. ' } ) , 500 )
2015-12-13 16:34:12 +07:00
elif jdata [ ' action ' ] == ' set_admin ' :
username = data [ ' username ' ]
2018-06-06 09:14:48 -02:00
if username == current_user . username :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' You cannot change you own admin rights. ' } ) , 400 )
2015-12-13 16:34:12 +07:00
is_admin = data [ ' is_admin ' ]
user = User ( username = username )
result = user . set_admin ( is_admin )
if result :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' Change user role of {0} ' . format ( username ) , created_by = current_user . username )
2015-12-13 16:34:12 +07:00
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Changed user role successfully. ' } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Cannot change user role. ' } ) , 500 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Action not supported. ' } ) , 400 )
except :
2018-03-31 08:21:02 +07:00
logging . error ( traceback . print_exc ( ) )
2015-12-13 16:34:12 +07:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' There is something wrong, please contact Administrator. ' } ) , 400 )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
@app.route ( ' /admin/account/edit/<account_name> ' , methods = [ ' GET ' , ' POST ' ] )
@app.route ( ' /admin/account/edit ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin_editaccount ( account_name = None ) :
2018-06-05 16:41:39 -02:00
users = User . query . all ( )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
if request . method == ' GET ' :
if account_name is None :
2018-06-05 16:41:39 -02:00
return render_template ( ' admin_editaccount.html ' , users = users , create = 1 )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
else :
account = Account . query . filter ( Account . name == account_name ) . first ( )
2018-06-05 16:41:39 -02:00
account_user_ids = account . get_user ( )
return render_template ( ' admin_editaccount.html ' , account = account , account_user_ids = account_user_ids , users = users , create = 0 )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
if request . method == ' POST ' :
fdata = request . form
2018-06-05 16:41:39 -02:00
new_user_list = request . form . getlist ( ' account_multi_user ' )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
2018-06-05 16:41:39 -02:00
# on POST, synthesize account and account_user_ids from form data
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
if not account_name :
account_name = fdata [ ' accountname ' ]
2018-06-05 16:41:39 -02:00
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
account = Account ( name = account_name , description = fdata [ ' accountdescription ' ] , contact = fdata [ ' accountcontact ' ] , mail = fdata [ ' accountmail ' ] )
2018-06-05 16:41:39 -02:00
account_user_ids = [ ]
for username in new_user_list :
userid = User ( username = username ) . get_user_info_by_username ( ) . id
account_user_ids . append ( userid )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
create = int ( fdata [ ' create ' ] )
if create :
# account __init__ sanitizes and lowercases the name, so to manage expectations
# we let the user reenter the name until it's not empty and it's valid (ignoring the case)
if account . name == " " or account . name != account_name . lower ( ) :
2018-06-05 16:41:39 -02:00
return render_template ( ' admin_editaccount.html ' , account = account , account_user_ids = account_user_ids , users = users , create = create , invalid_accountname = True )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
2018-06-05 16:41:39 -02:00
if Account . query . filter ( Account . name == account . name ) . first ( ) :
return render_template ( ' admin_editaccount.html ' , account = account , account_user_ids = account_user_ids , users = users , create = create , duplicate_accountname = True )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
result = account . create_account ( )
history = History ( msg = ' Create account {0} ' . format ( account . name ) , created_by = current_user . username )
else :
result = account . update_account ( )
history = History ( msg = ' Update account {0} ' . format ( account . name ) , created_by = current_user . username )
if result [ ' status ' ] :
2018-06-05 16:41:39 -02:00
account . grant_privileges ( new_user_list )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
history . add ( )
return redirect ( url_for ( ' admin_manageaccount ' ) )
2018-06-05 16:41:39 -02:00
return render_template ( ' admin_editaccount.html ' , account = account , account_user_ids = account_user_ids , users = users , create = create , error = result [ ' msg ' ] )
Initial support for Accounts
This adds initial support for accounts a concept meant to signify a customer, a department or any other entity that somehow owns or manages one or more domains.
The purpose is to be able to assign an account to any number of domains, making it easy to track who owns or manages a domain, significantly improving manageability in setups with a large number of domains.
An account consists of a mandatory, unique `name` and optional `description`, `contact` name and `mail` address. The account `name` is stripped of spaces and symbols, and lower cased before getting stored in the database and in PowerDNS, to help ensure some type of predictability and uniqueness in the database.
The term *account* is actually taken from the PowerDNS database, where the `domains.account` column is used to store the account relationship, in in the form of the account `name`.
The link to a domain in PowerDNS-Admin is done through the `domain.account_id` FOREIGN KEY, that is linked to the `account.id` PRIMARY KEY.
(cherry picked from commits 4e95f33dfb0676d1c401a033c28bca3be7d6ec26, da0d596bd019a339549e2c59630a8fdee65d0e22, 7f06e6aaf4fd8011c784f24b7bbbba5f52aef319, 1c624dad8749024033d1d15dd6242ca52b39f135)
2018-06-04 13:10:02 -02:00
@app.route ( ' /admin/manageaccount ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin_manageaccount ( ) :
if request . method == ' GET ' :
accounts = Account . query . order_by ( Account . name ) . all ( )
return render_template ( ' admin_manageaccount.html ' , accounts = accounts )
if request . method == ' POST ' :
#
# post data should in format
# {'action': 'delete_account', 'data': 'accountname'}
#
try :
jdata = request . json
data = jdata [ ' data ' ]
if jdata [ ' action ' ] == ' delete_account ' :
account = Account ( name = data )
result = account . delete_account ( )
if result :
history = History ( msg = ' Delete account {0} ' . format ( data ) , created_by = current_user . username )
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Account has been removed. ' } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Cannot remove account. ' } ) , 500 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Action not supported. ' } ) , 400 )
except :
logging . error ( traceback . print_exc ( ) )
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' There is something wrong, please contact Administrator. ' } ) , 400 )
2015-12-13 16:34:12 +07:00
@app.route ( ' /admin/history ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
@admin_role_required
def admin_history ( ) :
if request . method == ' POST ' :
h = History ( )
result = h . remove_all ( )
if result :
history = History ( msg = ' Remove all histories ' , created_by = current_user . username )
history . add ( )
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Changed user role successfully. ' } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Can not remove histories. ' } ) , 500 )
2016-08-05 16:20:41 +08:00
if request . method == ' GET ' :
2015-12-13 16:34:12 +07:00
histories = History . query . all ( )
return render_template ( ' admin_history.html ' , histories = histories )
2018-03-30 13:49:35 +07:00
2016-04-29 15:36:37 -06:00
@app.route ( ' /admin/settings ' , methods = [ ' GET ' ] )
@login_required
@admin_role_required
def admin_settings ( ) :
2016-08-05 16:20:41 +08:00
if request . method == ' GET ' :
Move setting definitions into code (rather than database).
For a setting to be useful, the code has to be able to make sense of it anyway. For this reason it makes sense, that the available settings are defined within the code, rather than in the database, where a missing row has previously caused problems. Instead, settings are now written to the database, when they are changed.
So instead of relying on the database initialization process to create all available settings for us in the database, the supported settings and their defaults are now in a `defaults` dict in the Setting class. With this in place, we can stop populating the `setting` table as a part of database initialization and it will be much easier to support new settings in the future (we no longer need to do anything to the database, to achieve that).
Another benefit is that any changes to default values will take effect automatically, unless the admin has already modified that setting to his/her liking.
To make it easier to get the value of a setting, falling back to defaults etc, a new function `get` has been added to the Setting class. Call it as `Setting().get('setting_name'), and it will take care of returning a setting from the database or return the default value for that setting, if nothing was found.
The `get` function returns `None`, if the setting passed to the function, does not exist in the `Setting.defaults` dict - Indicating that we don't know of a setting by that name.
2018-06-21 21:56:51 -02:00
# start with a copy of the setting defaults (ignore maintenance setting)
settings = Setting . defaults . copy ( )
settings . pop ( ' maintenance ' , None )
# update settings info with any customizations
for s in settings :
value = Setting ( ) . get ( s )
if value is not None :
settings [ s ] = value
2016-04-29 15:36:37 -06:00
return render_template ( ' admin_settings.html ' , settings = settings )
2016-08-05 16:20:41 +08:00
2018-03-30 13:49:35 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /admin/setting/<path:setting>/toggle ' , methods = [ ' POST ' ] )
2016-04-29 15:36:37 -06:00
@login_required
@admin_role_required
def admin_settings_toggle ( setting ) :
result = Setting ( ) . toggle ( setting )
if ( result ) :
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Toggled setting successfully. ' } ) , 200 )
else :
2016-05-15 12:47:02 -06:00
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Unable to toggle setting. ' } ) , 500 )
2015-12-13 16:34:12 +07:00
2018-03-30 13:49:35 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /admin/setting/<path:setting>/edit ' , methods = [ ' POST ' ] )
2016-06-08 19:23:08 -06:00
@login_required
@admin_role_required
def admin_settings_edit ( setting ) :
2018-03-30 13:49:35 +07:00
jdata = request . json
2016-06-08 19:23:08 -06:00
new_value = jdata [ ' value ' ]
result = Setting ( ) . set ( setting , new_value )
2018-03-30 13:49:35 +07:00
2016-06-08 19:23:08 -06:00
if ( result ) :
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Toggled setting successfully. ' } ) , 200 )
else :
return make_response ( jsonify ( { ' status ' : ' error ' , ' msg ' : ' Unable to toggle setting. ' } ) , 500 )
2018-03-30 13:49:35 +07:00
2015-12-16 14:21:30 +07:00
@app.route ( ' /user/profile ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
def user_profile ( ) :
2017-11-01 01:34:29 +01:00
external_account = False
2018-04-08 09:09:08 +07:00
if ' external_auth ' in session :
2017-11-01 01:34:29 +01:00
external_account = session [ ' external_auth ' ]
if request . method == ' GET ' or external_account :
return render_template ( ' user_profile.html ' , external_account = external_account )
2015-12-16 14:21:30 +07:00
if request . method == ' POST ' :
2015-12-17 00:50:28 +07:00
# get new profile info
2015-12-16 14:21:30 +07:00
firstname = request . form [ ' firstname ' ] if ' firstname ' in request . form else ' '
lastname = request . form [ ' lastname ' ] if ' lastname ' in request . form else ' '
email = request . form [ ' email ' ] if ' email ' in request . form else ' '
2015-12-17 00:50:28 +07:00
new_password = request . form [ ' password ' ] if ' password ' in request . form else ' '
2015-12-16 14:21:30 +07:00
2016-06-16 15:33:05 +07:00
# json data
if request . data :
2018-03-30 13:49:35 +07:00
jdata = request . json
2016-06-16 15:33:05 +07:00
data = jdata [ ' data ' ]
if jdata [ ' action ' ] == ' enable_otp ' :
enable_otp = data [ ' enable_otp ' ]
user = User ( username = current_user . username )
user . update_profile ( enable_otp = enable_otp )
2018-04-01 07:57:41 +07:00
return make_response ( jsonify ( { ' status ' : ' ok ' , ' msg ' : ' Change OTP Authentication successfully. Status: {0} ' . format ( enable_otp ) } ) , 200 )
2016-06-16 15:33:05 +07:00
2015-12-17 00:50:28 +07:00
# get new avatar
save_file_name = None
if ' file ' in request . files :
file = request . files [ ' file ' ]
if file :
filename = secure_filename ( file . filename )
file_extension = filename . rsplit ( ' . ' , 1 ) [ 1 ]
2016-08-05 16:20:41 +08:00
if file_extension . lower ( ) in [ ' jpg ' , ' jpeg ' , ' png ' ] :
save_file_name = current_user . username + ' . ' + file_extension
2015-12-17 00:50:28 +07:00
file . save ( os . path . join ( app . config [ ' UPLOAD_DIR ' ] , ' avatar ' , save_file_name ) )
# update user profile
user = User ( username = current_user . username , plain_text_password = new_password , firstname = firstname , lastname = lastname , email = email , avatar = save_file_name , reload_info = False )
2015-12-16 14:21:30 +07:00
user . update_profile ( )
2015-12-17 00:50:28 +07:00
2017-11-01 01:34:29 +01:00
return render_template ( ' user_profile.html ' , external_account = external_account )
2015-12-16 14:21:30 +07:00
2015-12-17 00:50:28 +07:00
2018-03-31 06:52:14 +07:00
@app.route ( ' /user/avatar/<path:filename> ' )
2015-12-17 00:50:28 +07:00
def user_avatar ( filename ) :
return send_from_directory ( os . path . join ( app . config [ ' UPLOAD_DIR ' ] , ' avatar ' ) , filename )
2016-06-16 15:33:05 +07:00
@app.route ( ' /qrcode ' )
@login_required
def qrcode ( ) :
if not current_user :
return redirect ( url_for ( ' index ' ) )
# render qrcode for FreeTOTP
2016-09-17 06:41:22 -07:00
img = qrc . make ( current_user . get_totp_uri ( ) , image_factory = qrc_svg . SvgImage )
2016-06-16 15:33:05 +07:00
stream = BytesIO ( )
2016-09-17 06:41:22 -07:00
img . save ( stream )
2016-06-16 15:33:05 +07:00
return stream . getvalue ( ) , 200 , {
' Content-Type ' : ' image/svg+xml ' ,
' Cache-Control ' : ' no-cache, no-store, must-revalidate ' ,
' Pragma ' : ' no-cache ' ,
' Expires ' : ' 0 ' }
2016-06-20 16:32:14 +07:00
@app.route ( ' /nic/checkip.html ' , methods = [ ' GET ' , ' POST ' ] )
def dyndns_checkip ( ) :
# route covers the default ddclient 'web' setting for the checkip service
2016-07-01 16:02:37 -06:00
return render_template ( ' dyndns.html ' , response = request . environ . get ( ' HTTP_X_REAL_IP ' , request . remote_addr ) )
2016-06-20 16:32:14 +07:00
2018-03-30 13:49:35 +07:00
2016-06-20 16:32:14 +07:00
@app.route ( ' /nic/update ' , methods = [ ' GET ' , ' POST ' ] )
@dyndns_login_required
def dyndns_update ( ) :
# dyndns protocol response codes in use are:
# good: update successful
# nochg: IP address already set to update address
# nohost: hostname does not exist for this user account
# 911: server error
# have to use 200 HTTP return codes because ddclient does not read the return string if the code is other than 200
# reference: https://help.dyn.com/remote-access-api/perform-update/
# reference: https://help.dyn.com/remote-access-api/return-codes/
hostname = request . args . get ( ' hostname ' )
myip = request . args . get ( ' myip ' )
try :
# get all domains owned by the current user
domains = User ( id = current_user . id ) . get_domain ( )
except :
return render_template ( ' dyndns.html ' , response = ' 911 ' ) , 200
2016-08-05 16:20:41 +08:00
2016-07-02 11:24:13 -06:00
domain = None
domain_segments = hostname . split ( ' . ' )
for index in range ( len ( domain_segments ) ) :
full_domain = ' . ' . join ( domain_segments )
potential_domain = Domain . query . filter ( Domain . name == full_domain ) . first ( )
if potential_domain in domains :
domain = potential_domain
break
2016-08-18 22:05:15 +02:00
domain_segments . pop ( 0 )
2016-08-05 16:20:41 +08:00
2016-07-02 11:24:13 -06:00
if not domain :
2018-04-01 07:57:41 +07:00
history = History ( msg = " DynDNS update: attempted update of {0} but it does not exist for this user " . format ( hostname ) , created_by = current_user . username )
2016-07-02 11:24:13 -06:00
history . add ( )
return render_template ( ' dyndns.html ' , response = ' nohost ' ) , 200
2016-08-05 16:20:41 +08:00
2016-07-02 11:24:13 -06:00
r = Record ( )
r . name = hostname
# check if the user requested record exists within this domain
2018-01-23 12:08:50 +03:00
if r . exists ( domain . name ) and r . is_allowed_edit ( ) :
2016-07-02 11:24:13 -06:00
if r . data == myip :
# record content did not change, return 'nochg'
2018-04-01 07:57:41 +07:00
history = History ( msg = " DynDNS update: attempted update of {0} but record did not change " . format ( hostname ) , created_by = current_user . username )
2016-07-02 11:24:13 -06:00
history . add ( )
return render_template ( ' dyndns.html ' , response = ' nochg ' ) , 200
else :
oldip = r . data
result = r . update ( domain . name , myip )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' DynDNS update: updated record {0} in zone {1} , it changed from {2} to {3} ' . format ( hostname , domain . name , oldip , myip ) , detail = str ( result ) , created_by = current_user . username )
2016-06-20 16:32:14 +07:00
history . add ( )
2016-07-02 11:24:13 -06:00
return render_template ( ' dyndns.html ' , response = ' good ' ) , 200
2016-06-20 16:32:14 +07:00
else :
2016-07-02 11:24:13 -06:00
return render_template ( ' dyndns.html ' , response = ' 911 ' ) , 200
2018-01-23 12:08:50 +03:00
elif r . is_allowed_edit ( ) :
2016-07-02 11:24:13 -06:00
ondemand_creation = DomainSetting . query . filter ( DomainSetting . domain == domain ) . filter ( DomainSetting . setting == ' create_via_dyndns ' ) . first ( )
2016-07-03 12:42:14 -06:00
if ( ondemand_creation != None ) and ( strtobool ( ondemand_creation . value ) == True ) :
2016-07-02 18:36:21 -06:00
record = Record ( name = hostname , type = ' A ' , data = myip , status = False , ttl = 3600 )
2016-07-02 11:24:13 -06:00
result = record . add ( domain . name )
if result [ ' status ' ] == ' ok ' :
2018-04-01 07:57:41 +07:00
history = History ( msg = ' DynDNS update: created record {0} in zone {1} , it now represents {2} ' . format ( hostname , domain . name , myip ) , detail = str ( result ) , created_by = current_user . username )
2016-07-02 11:24:13 -06:00
history . add ( )
return render_template ( ' dyndns.html ' , response = ' good ' ) , 200
2016-08-05 16:20:41 +08:00
2018-04-01 07:57:41 +07:00
history = History ( msg = ' DynDNS update: attempted update of {0} but it does not exist for this user ' . format ( hostname ) , created_by = current_user . username )
2016-06-20 16:32:14 +07:00
history . add ( )
return render_template ( ' dyndns.html ' , response = ' nohost ' ) , 200
2015-12-13 16:34:12 +07:00
@app.route ( ' / ' , methods = [ ' GET ' , ' POST ' ] )
@login_required
def index ( ) :
2016-08-05 16:20:41 +08:00
return redirect ( url_for ( ' dashboard ' ) )
2015-12-13 16:34:12 +07:00
# END VIEWS