├── PDNS_Manager_HL_Overview.png
├── PDNS_Manager_login_window.png
├── PDNS_Manager_records_window.png
├── PDNS_Manager_signup_window.png
├── res
├── css
│ ├── webfonts
│ │ ├── fa-brands-400.eot
│ │ ├── fa-brands-400.ttf
│ │ ├── fa-solid-900.eot
│ │ ├── fa-solid-900.ttf
│ │ ├── fa-solid-900.woff
│ │ ├── fa-brands-400.woff
│ │ ├── fa-brands-400.woff2
│ │ ├── fa-regular-400.eot
│ │ ├── fa-regular-400.ttf
│ │ ├── fa-regular-400.woff
│ │ ├── fa-regular-400.woff2
│ │ └── fa-solid-900.woff2
│ ├── toastr.min.css
│ └── awesome
│ │ └── all.min.css
└── js
│ ├── toastr.min.js
│ ├── api.js
│ └── toastr.js.map
├── api2
├── credentials-sample.php
├── settings-sample.php
├── auth_helper.php
├── index.php
└── pdns_helper.php
├── composer.json
├── LICENSE
├── README.md
└── index.html
/PDNS_Manager_HL_Overview.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/PDNS_Manager_HL_Overview.png
--------------------------------------------------------------------------------
/PDNS_Manager_login_window.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/PDNS_Manager_login_window.png
--------------------------------------------------------------------------------
/PDNS_Manager_records_window.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/PDNS_Manager_records_window.png
--------------------------------------------------------------------------------
/PDNS_Manager_signup_window.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/PDNS_Manager_signup_window.png
--------------------------------------------------------------------------------
/res/css/webfonts/fa-brands-400.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-brands-400.eot
--------------------------------------------------------------------------------
/res/css/webfonts/fa-brands-400.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-brands-400.ttf
--------------------------------------------------------------------------------
/res/css/webfonts/fa-solid-900.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-solid-900.eot
--------------------------------------------------------------------------------
/res/css/webfonts/fa-solid-900.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-solid-900.ttf
--------------------------------------------------------------------------------
/res/css/webfonts/fa-solid-900.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-solid-900.woff
--------------------------------------------------------------------------------
/res/css/webfonts/fa-brands-400.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-brands-400.woff
--------------------------------------------------------------------------------
/res/css/webfonts/fa-brands-400.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-brands-400.woff2
--------------------------------------------------------------------------------
/res/css/webfonts/fa-regular-400.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-regular-400.eot
--------------------------------------------------------------------------------
/res/css/webfonts/fa-regular-400.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-regular-400.ttf
--------------------------------------------------------------------------------
/res/css/webfonts/fa-regular-400.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-regular-400.woff
--------------------------------------------------------------------------------
/res/css/webfonts/fa-regular-400.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-regular-400.woff2
--------------------------------------------------------------------------------
/res/css/webfonts/fa-solid-900.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/vbeskrovny/PDNS-Manager/HEAD/res/css/webfonts/fa-solid-900.woff2
--------------------------------------------------------------------------------
/api2/credentials-sample.php:
--------------------------------------------------------------------------------
1 | array('password' => 'some_md5_password_hash', 'secret' => 'some_otp_secret'),
7 |
8 |
9 |
10 | ]);
11 |
12 |
13 |
14 | define('DDNS_TOKENS',
15 | [
16 | 'some_token' => 'some_token_description',
17 |
18 |
19 | ]);
20 |
21 |
--------------------------------------------------------------------------------
/composer.json:
--------------------------------------------------------------------------------
1 | {
2 | "require": {
3 | "guzzlehttp/guzzle": "^7.0",
4 | "slim/slim": "4.*",
5 | "slim/psr7": "^1.3",
6 | "nyholm/psr7": "^1.3",
7 | "nyholm/psr7-server": "^1.0",
8 | "guzzlehttp/psr7": "^1.7",
9 | "http-interop/http-factory-guzzle": "^1.0",
10 | "laminas/laminas-diactoros": "^2.4",
11 | "defuse/php-encryption": "^2.2",
12 | "spomky-labs/otphp": "^8.3",
13 | "endroid/qr-code": "^3.9"
14 | }
15 | }
16 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2021 Ben
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/api2/settings-sample.php:
--------------------------------------------------------------------------------
1 | 3x AUTH_INTERVAL (in seconds)
23 | define('AUTH_LIFE', 900);
24 |
25 |
26 | ## Replace with your own by running: php -f ./auth_helper.php
27 | define('AUTH_KEY', 'def0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000936a8ed840e6b8cdcf2c41e89331dabf7b1a13b51bcc93133f5ff5d31a36fe5146720e4390c9e98376e1ade741dc7dd8834d3cb6ea98b7bd8a29c6ef315ea42f');
28 |
29 |
30 | ## PDNS Api key
31 | define('API_KEY', '00000000000000000000000000000000');
32 |
33 |
34 | define('ZONE_DEFAULTS', [
35 | 'dns' => 'dns1.mydns.com.',
36 | 'hostmaster' => 'hostmaster.mydns.com.',
37 | 'nameservers' => ['dns1.mydns.com.', 'dns2.mydns.com.'],
38 | 'masters' => ['dns1.mydns.com.'],
39 | 'refresh' => 14400,
40 | 'retry' => 3600,
41 | 'expire' => 604800,
42 | 'ttl' => 3600,
43 | ]);
44 |
45 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # PDNS-Manager
2 | Lightweight PowerDNS management frontend
3 |
4 |
5 | ### Intention and prerequisites
6 | 1. During my IT management responsibilities I found some time to contribute to our team of engineers and company overall.
7 | 2. Tired of looking for a decent PowerDNS frontend manager -> let's create our own.
8 | 3. Should be lightweight, dead simple and could be easily modified if needed.
9 | 4. No need for the 3rd party storage databases (MySQL, PostgreSQL, e.t.c. databases), complex frameworks -> direct communication with the PowerDNS using API
10 | 5. Use PHP, HTML and JavaScript (jQuery) only
11 | 6. Some extra security on top (OTP)
12 |
13 | - - - -
14 |
15 |
16 | ### What works
17 | 1. Adding and removing zones
18 | 2. Modifying records
19 | 3. Adding records (A, TXT, CNAME, MX, SRV)
20 | 4. DDNS minimal usage (check Wiki as well):
21 | * Add: https://pdns.yourdns.com/pdns/ddns/token=aabbcc/name=myhost.example.com[/content=1.2.3.4[/keep=1|0]
22 | * Remove: https://pdns.yourdns.com/pdns/ddns/token=aabbcc/name=myhost.example.com/content=null
23 |
24 | - - - -
25 |
26 |
27 | ### Requirements
28 | 1. PowerDNS
29 | 2. Web server
30 | 3. PHP
31 |
32 | - - - -
33 |
34 | ### Tested on
35 | 1. PHP-FPM: 7.2.4
36 | 2. Ubuntu: 18.04.5
37 | 3. NGINX: 1.14.0
38 |
39 | - - - -
40 |
41 | ### Disclaimer
42 | 1. Use at your own risk
43 | 2. No input data has been validated (apart from adding . (dot) at the end of zones/records), so be careful
44 | 3. If unsure -> add https + basic auth in front of PDNS Manager directory
45 | 4. Feel free to contribute, correct the bugs, add extra functionality
46 |
47 | - - - -
48 |
49 | ### 2DO
50 | - [x] DDNS (via GET)
51 | - [x] DDNS (via POST)
52 | - [x] DDNS get IP from source request by default
53 | - [x] OTP can be turned off by setting secret to 'null' value: e.g. 'user' => array('password' => 'hash', 'secret' => null); and enabled/disabled via settings
54 | - [ ] Protect credentials/token against bruteforcing
55 | - [ ] Validate TXT record to have surrounding quotes
56 | - [ ] Implement TOKEN -> ZONE / RECORD policy
57 | - [ ] Implement USER -> ZONE / RECORD policy
58 | - [ ] Multi-user support (credentials, zones, records)
59 | - [ ] Validate content depending on the record type
60 | - [ ] Pass status codes from PowerDNS API back to application API
61 | - [ ] Cleanup
62 | - [ ] PDNS_Helper -> prepare() -> array
63 | - [ ] Minify everything
64 | - [ ] Possibility to edit settings from the GUI (???)
65 | - [ ] Possibility to edit credentials from the GUI (???)
66 |
67 | - - - -
68 |
69 | ### High level logic overview
70 | 
71 |
72 | - - - -
73 |
74 | ### Screenshots
75 | #### Sign in window
76 | 
77 |
78 | - - - -
79 |
80 | #### Sign up window
81 | 
82 |
83 | - - - -
84 |
85 | #### Records editing window
86 | 
87 |
88 | - - - -
89 |
90 | ### Setup
91 | 1. Clone the project
92 | 2. Configure the web server (NGINX snippet) + enable PHP
93 | ```
94 | location /pdns/api2 {
95 | root /var/www/pdns.yourdns.com/pdns/api2;
96 | try_files $uri /pdns/api2/index.php$is_args$args;
97 | }
98 |
99 | location / {
100 | try_files $uri $uri/ =404;
101 | }
102 | ```
103 | 3. Fetch dependencies using composer (check composer.json to see what is needed)
104 | 4. Navigate to https://pdns.yourdns.com -> you should see the login prompt and be able to "sign up"
105 | 5. Save the received credentials to the '-sample.php' files and rename the files (credentials-sample.php and settings-sample.php) by removing '-sample'
106 | 6. Enjoy...
107 |
--------------------------------------------------------------------------------
/api2/auth_helper.php:
--------------------------------------------------------------------------------
1 | 0) {
11 | require __DIR__ . '/../vendor/autoload.php';
12 | printf("\nNew AUTH_KEY: %s\n\n", Key::createNewRandomKey()->saveToAsciiSafeString());
13 | exit;
14 | }
15 |
16 |
17 |
18 | class AUTH_Helper {
19 |
20 | private $key;
21 |
22 | function __construct() {
23 | $this->key = Key::loadFromAsciiSafeString(AUTH_KEY);
24 | }
25 |
26 |
27 | function test($something = null) {
28 |
29 | return $something;
30 |
31 | }
32 |
33 |
34 | function generate_key() {
35 | $key = Key::createNewRandomKey();
36 | return $key->saveToAsciiSafeString();
37 | }
38 |
39 |
40 | function encrypt($text) {
41 | return Crypto::encrypt(sprintf('%s', $text), $this->key, $raw_binary = false);
42 | }
43 |
44 | function decrypt($text) {
45 | return Crypto::decrypt(sprintf('%s', $text), $this->key, $raw_binary = false);
46 | }
47 |
48 |
49 | function check_auth($request) {
50 |
51 | $auth_status = false;
52 |
53 |
54 |
55 | $cookies = new \Slim\Psr7\Cookies($request->getCookieParams());
56 | $auth_cookie = $cookies->get('pdns_auth_cookie');
57 |
58 | if ($auth_cookie) {
59 |
60 | $auth_cookie_ary = json_decode($this->decrypt($auth_cookie), true);
61 |
62 |
63 | $ts = $auth_cookie_ary['ts'];
64 | $username = $auth_cookie_ary['username'];
65 | $password = $auth_cookie_ary['password'];
66 |
67 |
68 | ## Check TS
69 | if (($ts + AUTH_LIFE + 1) >= time()) { ## TS is valid
70 |
71 | if (array_key_exists($username, AUTH_HASH) && AUTH_HASH[$username]['password'] == md5($password)) {
72 |
73 | $auth_status = true;
74 |
75 | $auth_cookie =
76 | $this->encrypt(
77 | json_encode([
78 | 'ts' => time(),
79 | 'username' => $username,
80 | 'password' => $password
81 | ])
82 | );
83 |
84 | } else {
85 |
86 | $auth_cookie = null;
87 |
88 | }
89 |
90 |
91 | } else {
92 |
93 | $auth_cookie = null;
94 |
95 | }
96 |
97 |
98 |
99 | } else {
100 |
101 | $auth_cookie = null;
102 |
103 | }
104 |
105 |
106 |
107 |
108 | return [ $auth_status, $auth_cookie, 'auth_status' => $auth_status, 'auth_cookie' => $auth_cookie ];
109 |
110 | }
111 |
112 |
113 |
114 | function is_totp_ok($username, $otp) {
115 | if (OTP_ENABLED) {
116 | if (array_key_exists('secret', AUTH_HASH[$username])) {
117 | if (AUTH_HASH[$username]['secret'] === null) {
118 | return true;
119 | } else {
120 | if (class_exists('OTPHP\TOTP')) {
121 | $totp = new \OTPHP\TOTP(null, AUTH_HASH[$username]['secret']);
122 | return $totp->verify($otp);
123 | } else {
124 | return false;
125 | }
126 | }
127 | } else {
128 | return true;
129 | }
130 | } else {
131 | return true;
132 | }
133 | }
134 |
135 |
136 | function signup($username) {
137 |
138 | if (OTP_ENABLED === true && class_exists('OTPHP\TOTP')) {
139 |
140 | $totp = new \OTPHP\TOTP();
141 |
142 | $otp_secret = $totp->getSecret();
143 | $totp = new \OTPHP\TOTP($username, $otp_secret);
144 | $otp_url = $totp->getProvisioningUri();
145 |
146 |
147 | $qrCode = new QrCode($otp_url);
148 | $otp_qr = $qrCode->writeDataUri();
149 |
150 |
151 | return [ $otp_secret, $otp_url, $otp_qr ];
152 |
153 | } else {
154 |
155 | return [ 'TOTP disabled and/or not installed', 'TOTP disabled and/or not installed', null ];
156 |
157 | }
158 |
159 | }
160 |
161 |
162 |
163 | function do_auth($username, $password, $otp) {
164 |
165 | $auth_status = false;
166 | $auth_cookie = null;
167 |
168 | if (array_key_exists($username, AUTH_HASH) && AUTH_HASH[$username]['password'] == md5($password) && $this->is_totp_ok($username, $otp) === true) {
169 |
170 | $auth_status = true;
171 |
172 | $auth_cookie =
173 | $this->encrypt(
174 | json_encode([
175 | 'ts' => time(),
176 | 'username' => $username,
177 | 'password' => $password
178 | ])
179 | );
180 |
181 | }
182 |
183 |
184 | return [ $auth_status, $auth_cookie ];
185 |
186 | }
187 |
188 |
189 | }
190 |
191 |
192 |
--------------------------------------------------------------------------------
/res/js/toastr.min.js:
--------------------------------------------------------------------------------
1 | !function(e){e(["jquery"],function(e){return function(){function t(e,t,n){return g({type:O.error,iconClass:m().iconClasses.error,message:e,optionsOverride:n,title:t})}function n(t,n){return t||(t=m()),v=e("#"+t.containerId),v.length?v:(n&&(v=d(t)),v)}function o(e,t,n){return g({type:O.info,iconClass:m().iconClasses.info,message:e,optionsOverride:n,title:t})}function s(e){C=e}function i(e,t,n){return g({type:O.success,iconClass:m().iconClasses.success,message:e,optionsOverride:n,title:t})}function a(e,t,n){return g({type:O.warning,iconClass:m().iconClasses.warning,message:e,optionsOverride:n,title:t})}function r(e,t){var o=m();v||n(o),u(e,o,t)||l(o)}function c(t){var o=m();return v||n(o),t&&0===e(":focus",t).length?void h(t):void(v.children().length&&v.remove())}function l(t){for(var n=v.children(),o=n.length-1;o>=0;o--)u(e(n[o]),t)}function u(t,n,o){var s=!(!o||!o.force)&&o.force;return!(!t||!s&&0!==e(":focus",t).length)&&(t[n.hideMethod]({duration:n.hideDuration,easing:n.hideEasing,complete:function(){h(t)}}),!0)}function d(t){return v=e("
").attr("id",t.containerId).addClass(t.positionClass),v.appendTo(e(t.target)),v}function p(){return{tapToDismiss:!0,toastClass:"toast",containerId:"toast-container",debug:!1,showMethod:"fadeIn",showDuration:300,showEasing:"swing",onShown:void 0,hideMethod:"fadeOut",hideDuration:1e3,hideEasing:"swing",onHidden:void 0,closeMethod:!1,closeDuration:!1,closeEasing:!1,closeOnHover:!0,extendedTimeOut:1e3,iconClasses:{error:"toast-error",info:"toast-info",success:"toast-success",warning:"toast-warning"},iconClass:"toast-info",positionClass:"toast-top-right",timeOut:5e3,titleClass:"toast-title",messageClass:"toast-message",escapeHtml:!1,target:"body",closeHtml:'',closeClass:"toast-close-button",newestOnTop:!0,preventDuplicates:!1,progressBar:!1,progressClass:"toast-progress",rtl:!1}}function f(e){C&&C(e)}function g(t){function o(e){return null==e&&(e=""),e.replace(/&/g,"&").replace(/"/g,""").replace(/'/g,"'").replace(//g,">")}function s(){c(),u(),d(),p(),g(),C(),l(),i()}function i(){var e="";switch(t.iconClass){case"toast-success":case"toast-info":e="polite";break;default:e="assertive"}I.attr("aria-live",e)}function a(){E.closeOnHover&&I.hover(H,D),!E.onclick&&E.tapToDismiss&&I.click(b),E.closeButton&&j&&j.click(function(e){e.stopPropagation?e.stopPropagation():void 0!==e.cancelBubble&&e.cancelBubble!==!0&&(e.cancelBubble=!0),E.onCloseClick&&E.onCloseClick(e),b(!0)}),E.onclick&&I.click(function(e){E.onclick(e),b()})}function r(){I.hide(),I[E.showMethod]({duration:E.showDuration,easing:E.showEasing,complete:E.onShown}),E.timeOut>0&&(k=setTimeout(b,E.timeOut),F.maxHideTime=parseFloat(E.timeOut),F.hideEta=(new Date).getTime()+F.maxHideTime,E.progressBar&&(F.intervalId=setInterval(x,10)))}function c(){t.iconClass&&I.addClass(E.toastClass).addClass(y)}function l(){E.newestOnTop?v.prepend(I):v.append(I)}function u(){if(t.title){var e=t.title;E.escapeHtml&&(e=o(t.title)),M.append(e).addClass(E.titleClass),I.append(M)}}function d(){if(t.message){var e=t.message;E.escapeHtml&&(e=o(t.message)),B.append(e).addClass(E.messageClass),I.append(B)}}function p(){E.closeButton&&(j.addClass(E.closeClass).attr("role","button"),I.prepend(j))}function g(){E.progressBar&&(q.addClass(E.progressClass),I.prepend(q))}function C(){E.rtl&&I.addClass("rtl")}function O(e,t){if(e.preventDuplicates){if(t.message===w)return!0;w=t.message}return!1}function b(t){var n=t&&E.closeMethod!==!1?E.closeMethod:E.hideMethod,o=t&&E.closeDuration!==!1?E.closeDuration:E.hideDuration,s=t&&E.closeEasing!==!1?E.closeEasing:E.hideEasing;if(!e(":focus",I).length||t)return clearTimeout(F.intervalId),I[n]({duration:o,easing:s,complete:function(){h(I),clearTimeout(k),E.onHidden&&"hidden"!==P.state&&E.onHidden(),P.state="hidden",P.endTime=new Date,f(P)}})}function D(){(E.timeOut>0||E.extendedTimeOut>0)&&(k=setTimeout(b,E.extendedTimeOut),F.maxHideTime=parseFloat(E.extendedTimeOut),F.hideEta=(new Date).getTime()+F.maxHideTime)}function H(){clearTimeout(k),F.hideEta=0,I.stop(!0,!0)[E.showMethod]({duration:E.showDuration,easing:E.showEasing})}function x(){var e=(F.hideEta-(new Date).getTime())/F.maxHideTime*100;q.width(e+"%")}var E=m(),y=t.iconClass||E.iconClass;if("undefined"!=typeof t.optionsOverride&&(E=e.extend(E,t.optionsOverride),y=t.optionsOverride.iconClass||y),!O(E,t)){T++,v=n(E,!0);var k=null,I=e(""),M=e(""),B=e(""),q=e(""),j=e(E.closeHtml),F={intervalId:null,hideEta:null,maxHideTime:null},P={toastId:T,state:"visible",startTime:new Date,options:E,map:t};return s(),r(),a(),f(P),E.debug&&console&&console.log(P),I}}function m(){return e.extend({},p(),b.options)}function h(e){v||(v=n()),e.is(":visible")||(e.remove(),e=null,0===v.children().length&&(v.remove(),w=void 0))}var v,C,w,T=0,O={error:"error",info:"info",success:"success",warning:"warning"},b={clear:r,remove:c,error:t,getContainer:n,info:o,options:{},subscribe:s,success:i,version:"2.1.4",warning:a};return b}()})}("function"==typeof define&&define.amd?define:function(e,t){"undefined"!=typeof module&&module.exports?module.exports=t(require("jquery")):window.toastr=t(window.jQuery)});
2 | //# sourceMappingURL=toastr.js.map
3 |
--------------------------------------------------------------------------------
/res/css/toastr.min.css:
--------------------------------------------------------------------------------
1 | .toast-title{font-weight:700}.toast-message{-ms-word-wrap:break-word;word-wrap:break-word}.toast-message a,.toast-message label{color:#FFF}.toast-message a:hover{color:#CCC;text-decoration:none}.toast-close-button{position:relative;right:-.3em;top:-.3em;float:right;font-size:20px;font-weight:700;color:#FFF;-webkit-text-shadow:0 1px 0 #fff;text-shadow:0 1px 0 #fff;opacity:.8;-ms-filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=80);filter:alpha(opacity=80);line-height:1}.toast-close-button:focus,.toast-close-button:hover{color:#000;text-decoration:none;cursor:pointer;opacity:.4;-ms-filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=40);filter:alpha(opacity=40)}.rtl .toast-close-button{left:-.3em;float:left;right:.3em}button.toast-close-button{padding:0;cursor:pointer;background:0 0;border:0;-webkit-appearance:none}.toast-top-center{top:0;right:0;width:100%}.toast-bottom-center{bottom:0;right:0;width:100%}.toast-top-full-width{top:0;right:0;width:100%}.toast-bottom-full-width{bottom:0;right:0;width:100%}.toast-top-left{top:12px;left:12px}.toast-top-right{top:12px;right:12px}.toast-bottom-right{right:12px;bottom:12px}.toast-bottom-left{bottom:12px;left:12px}#toast-container{position:fixed;z-index:999999;pointer-events:none}#toast-container *{-moz-box-sizing:border-box;-webkit-box-sizing:border-box;box-sizing:border-box}#toast-container>div{position:relative;pointer-events:auto;overflow:hidden;margin:0 0 6px;padding:15px 15px 15px 50px;width:300px;-moz-border-radius:3px;-webkit-border-radius:3px;border-radius:3px;background-position:15px center;background-repeat:no-repeat;-moz-box-shadow:0 0 12px #999;-webkit-box-shadow:0 0 12px #999;box-shadow:0 0 12px #999;color:#FFF;opacity:.8;-ms-filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=80);filter:alpha(opacity=80)}#toast-container>div.rtl{direction:rtl;padding:15px 50px 15px 15px;background-position:right 15px center}#toast-container>div:hover{-moz-box-shadow:0 0 12px #000;-webkit-box-shadow:0 0 12px #000;box-shadow:0 0 12px #000;opacity:1;-ms-filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);filter:alpha(opacity=100);cursor:pointer}#toast-container>.toast-info{background-image:url()!important}#toast-container>.toast-error{background-image:url()!important}#toast-container>.toast-success{background-image:url()!important}#toast-container>.toast-warning{background-image:url()!important}#toast-container.toast-bottom-center>div,#toast-container.toast-top-center>div{width:300px;margin-left:auto;margin-right:auto}#toast-container.toast-bottom-full-width>div,#toast-container.toast-top-full-width>div{width:96%;margin-left:auto;margin-right:auto}.toast{background-color:#030303}.toast-success{background-color:#51A351}.toast-error{background-color:#BD362F}.toast-info{background-color:#2F96B4}.toast-warning{background-color:#F89406}.toast-progress{position:absolute;left:0;bottom:0;height:4px;background-color:#000;opacity:.4;-ms-filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=40);filter:alpha(opacity=40)}@media all and (max-width:240px){#toast-container>div{padding:8px 8px 8px 50px;width:11em}#toast-container>div.rtl{padding:8px 50px 8px 8px}#toast-container .toast-close-button{right:-.2em;top:-.2em}#toast-container .rtl .toast-close-button{left:-.2em;right:.2em}}@media all and (min-width:241px) and (max-width:480px){#toast-container>div{padding:8px 8px 8px 50px;width:18em}#toast-container>div.rtl{padding:8px 50px 8px 8px}#toast-container .toast-close-button{right:-.2em;top:-.2em}#toast-container .rtl .toast-close-button{left:-.2em;right:.2em}}@media all and (min-width:481px) and (max-width:768px){#toast-container>div{padding:15px 15px 15px 50px;width:25em}#toast-container>div.rtl{padding:15px 50px 15px 15px}}
--------------------------------------------------------------------------------
/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | PDNS Manager
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
79 |
80 |
81 |
82 |
83 |
84 |
124 |
125 |
126 |
127 |
128 |
158 |
159 |
160 |
161 |
162 |
163 |
164 |
267 |
268 |
269 |
270 |
--------------------------------------------------------------------------------
/api2/index.php:
--------------------------------------------------------------------------------
1 | setBasePath(URL_PREFIX);
23 |
24 |
25 | $app->get('/test[/{something}]', function (Request $request, Response $response, $args) {
26 |
27 | global $PDNS, $AUTH;
28 |
29 |
30 | // $payload = json_encode([$AUTH->test($args['something'])]);
31 | $payload = json_encode($AUTH->check_auth($request)['auth_status']);
32 | // $payload = json_encode($AUTH->generate_key());
33 |
34 |
35 | $response->getBody()->write($payload);
36 | return $response->withHeader('Content-Type', 'application/json');
37 |
38 | });
39 |
40 |
41 |
42 |
43 |
44 | $app->post('/signup', function (Request $request, Response $response) {
45 |
46 | global $AUTH;
47 |
48 | $params = $request->getParsedBody();
49 |
50 |
51 | $password_hash = md5($params['password']);
52 | list($otp_secret, $otp_url, $otp_qr) = $AUTH->signup($params['username']);
53 |
54 |
55 | $payload = json_encode(['password_hash' => $password_hash, 'otp_secret' => $otp_secret, 'otp_url' => $otp_url, 'otp_qr' => $otp_qr ]);
56 |
57 |
58 | $response->getBody()->write($payload);
59 | return $response->withHeader('Content-Type', 'application/json');
60 |
61 |
62 | });
63 |
64 |
65 |
66 |
67 | $app->get('/check_auth', function (Request $request, Response $response, $args) {
68 |
69 | global $AUTH;
70 |
71 |
72 | list($auth_status, $auth_cookie) = $AUTH->check_auth($request);
73 |
74 |
75 | $payload = json_encode(['auth_status' => $auth_status, 'auth_cookie' => $auth_cookie]);
76 |
77 |
78 |
79 |
80 | $response->getBody()->write($payload);
81 | return $response->withHeader('Content-Type', 'application/json');
82 |
83 |
84 | });
85 |
86 |
87 |
88 | $app->post('/do_auth', function (Request $request, Response $response) {
89 |
90 | global $AUTH;
91 |
92 | $params = $request->getParsedBody();
93 |
94 | $auth_status = false;
95 | $auth_cookie = null;
96 |
97 | if (array_key_exists('username', $params) && array_key_exists('password', $params) && array_key_exists('otp', $params)) {
98 |
99 | list($auth_status, $auth_cookie) = $AUTH->do_auth($params['username'], $params['password'], $params['otp']);
100 |
101 | }
102 |
103 |
104 |
105 | $payload = json_encode(['auth_status' => $auth_status, 'auth_cookie' => $auth_cookie]);
106 |
107 |
108 | $response->getBody()->write($payload);
109 | return $response->withHeader('Content-Type', 'application/json');
110 |
111 |
112 | });
113 |
114 |
115 |
116 | ## DDNS via POST
117 | $app->post('/ddns', function (Request $request, Response $response) {
118 |
119 | global $PDNS, $AUTH;
120 |
121 | $params = $request->getParsedBody();
122 |
123 |
124 | $status = $PDNS->do_ddns($params);
125 |
126 |
127 | $response->getBody()->write($status);
128 | return $response->withHeader('Content-Type', 'text/plain');
129 |
130 | });
131 |
132 |
133 | ## DDNS via GET
134 | $app->get('/ddns[/{get_params:.*}]', function (Request $request, Response $response, $args) {
135 |
136 | global $PDNS, $AUTH;
137 |
138 | $params = array();
139 | $get_params = explode('/', $args['get_params']);
140 |
141 |
142 | foreach ($get_params as $kv_pair) {
143 | if (preg_match('/^(.+)=(.+)$/', $kv_pair, $kv_ary)) {
144 | $key = $kv_ary[1];
145 | $val = $kv_ary[2];
146 | $params[$key] = $val;
147 | }
148 | }
149 |
150 |
151 | $status = $PDNS->do_ddns($params);
152 |
153 | $response->getBody()->write($status);
154 | return $response->withHeader('Content-Type', 'text/plain');
155 |
156 |
157 | });
158 |
159 |
160 |
161 | $app->post('/globals_init', function (Request $request, Response $response) {
162 |
163 | global $PDNS, $AUTH;
164 |
165 | $settings = array();
166 |
167 |
168 | if ($AUTH->check_auth($request)['auth_status'] === true) {
169 |
170 | $settings['AUTH_INTERVAL'] = AUTH_INTERVAL;
171 | $settings['ZONE_DEFAULTS'] = ZONE_DEFAULTS;
172 |
173 | }
174 |
175 |
176 | $payload = json_encode($settings);
177 |
178 |
179 | $response->getBody()->write($payload);
180 | return $response->withHeader('Content-Type', 'application/json');
181 |
182 |
183 | });
184 |
185 |
186 |
187 | $app->post('/remove_record', function (Request $request, Response $response) {
188 |
189 | global $PDNS, $AUTH;
190 |
191 | if ($AUTH->check_auth($request)['auth_status'] === true) {
192 |
193 | $params = $request->getParsedBody();
194 |
195 | $PDNS->remove_record($params['zone'], $params['type'], $params['name']);
196 |
197 |
198 | $payload = json_encode(['default' => true]);
199 |
200 | } else {
201 |
202 | $payload = json_encode(['auth_status' => false]);
203 |
204 | }
205 |
206 |
207 | $response->getBody()->write($payload);
208 | return $response->withHeader('Content-Type', 'application/json');
209 |
210 |
211 | });
212 |
213 |
214 |
215 | $app->post('/save_records', function (Request $request, Response $response) {
216 |
217 | global $PDNS, $AUTH;
218 |
219 | if ($AUTH->check_auth($request)['auth_status'] === true) {
220 |
221 | $params = $request->getParsedBody();
222 |
223 | $PDNS->save_records($params);
224 |
225 |
226 | $payload = json_encode(['default' => true]);
227 |
228 | } else {
229 |
230 | $payload = json_encode(['auth_status' => false]);
231 |
232 | }
233 |
234 |
235 | $response->getBody()->write($payload);
236 | return $response->withHeader('Content-Type', 'application/json');
237 |
238 |
239 | });
240 |
241 |
242 |
243 | $app->post('/remove_zone/{zone}', function (Request $request, Response $response, $args) {
244 |
245 | global $PDNS, $AUTH;
246 |
247 |
248 | if ($AUTH->check_auth($request)['auth_status'] === true) {
249 |
250 | $PDNS->remove_zone($args['zone']);
251 |
252 |
253 | $payload = json_encode(['default' => true]);
254 |
255 | } else {
256 |
257 | $payload = json_encode(['auth_status' => false]);
258 |
259 | }
260 |
261 |
262 | $response->getBody()->write($payload);
263 | return $response->withHeader('Content-Type', 'application/json');
264 |
265 | });
266 |
267 |
268 |
269 |
270 | $app->post('/add_zone', function (Request $request, Response $response) {
271 |
272 | global $PDNS, $AUTH;
273 |
274 |
275 | if ($AUTH->check_auth($request)['auth_status'] === true) {
276 |
277 | $params = $request->getParsedBody();
278 |
279 | $PDNS->add_zone($params);
280 |
281 | $payload = json_encode(['default' => true]);
282 |
283 | } else {
284 |
285 | $payload = json_encode(['auth_status' => false]);
286 |
287 | }
288 |
289 |
290 | $response->getBody()->write($payload);
291 | return $response->withHeader('Content-Type', 'application/json');
292 |
293 |
294 | });
295 |
296 |
297 | $app->post('/get_zones', function (Request $request, Response $response, $args) {
298 |
299 | global $PDNS, $AUTH;
300 |
301 |
302 | if ($AUTH->check_auth($request)['auth_status'] === true) {
303 |
304 | $payload = $PDNS->get_zones();
305 |
306 | } else {
307 |
308 | $payload = json_encode(['auth_status' => false]);
309 |
310 | }
311 |
312 |
313 | $response->getBody()->write($payload);
314 | return $response->withHeader('Content-Type', 'application/json');
315 |
316 | });
317 |
318 |
319 |
320 | $app->post('/get_records/{zone}', function (Request $request, Response $response, $args) {
321 |
322 | global $PDNS, $AUTH;
323 |
324 | if ($AUTH->check_auth($request)['auth_status'] === true) {
325 |
326 | $payload = $PDNS->get_records($args['zone']);
327 |
328 | } else {
329 |
330 | $payload = json_encode(['auth_status' => false]);
331 |
332 | }
333 |
334 |
335 | $response->getBody()->write($payload);
336 | return $response->withHeader('Content-Type', 'application/json');
337 |
338 | });
339 |
340 |
341 |
342 | // DEFAULT - catch all
343 | $app->get('/{path:.*}', function ($request, $response, array $args) {
344 | $payload = json_encode(['default' => true]);
345 | $response->getBody()->write($payload);
346 | return $response->withHeader('Content-Type', 'application/json');
347 | });
348 |
349 |
350 |
351 |
352 | $app->run();
--------------------------------------------------------------------------------
/res/js/api.js:
--------------------------------------------------------------------------------
1 |
2 | var GLOBALS = {};
3 | var row_id = 1;
4 |
5 |
6 | function globals_init() {
7 |
8 | $.post('api2/globals_init', function(data) {
9 | GLOBALS = data;
10 | });
11 |
12 | }
13 |
14 |
15 | function signup() {
16 |
17 |
18 | var form_data = $('#auth_form').serializeArray();
19 | var all_valid = true;
20 |
21 | for (let key in form_data) {
22 |
23 | if (form_data[key].name == 'username') {
24 | if (form_data[key].value == '') {
25 | all_valid = false;
26 | $('.auth-form-username').toggleClass('is-invalid', true);
27 | } else {
28 | $('.auth-form-username').removeClass('is-invalid');
29 | }
30 | } else if (form_data[key].name == 'password') {
31 | if (form_data[key].value == '') {
32 | all_valid = false;
33 | $('.auth-form-password').toggleClass('is-invalid', true);
34 | } else {
35 | $('.auth-form-password').removeClass('is-invalid');
36 | }
37 | }
38 |
39 |
40 | }
41 |
42 |
43 | $('.auth-form-otp').removeClass('is-invalid');
44 |
45 |
46 | if (all_valid) {
47 |
48 | $.post('api2/signup', form_data, function(data) {
49 |
50 | $('input#passwordHash').val(data.password_hash);
51 | $('input#otpSecret').val(data.otp_secret);
52 | $('input#otpURL').val(data.otp_url);
53 | $('img#otpIMG').attr('src', data.otp_qr);
54 |
55 |
56 | $('.signup-div').show();
57 |
58 | });
59 |
60 | }
61 |
62 | }
63 |
64 |
65 |
66 | function update_cookie(content) {
67 | Cookies.set('pdns_auth_cookie', content, { secure: true, expires: 365 });
68 | }
69 |
70 |
71 | function remove_cookie() {
72 | Cookies.remove('pdns_auth_cookie', { secure: true });
73 | }
74 |
75 |
76 |
77 | function sign_out() {
78 | remove_cookie();
79 | location.reload();
80 | }
81 |
82 |
83 |
84 | function auth_loop() {
85 |
86 | var delay = GLOBALS['AUTH_INTERVAL'] * 1000;
87 |
88 | setTimeout( function() {
89 |
90 | $.get('api2/check_auth', function(data) {
91 |
92 | if (data.auth_status) {
93 |
94 | update_cookie(data.auth_cookie);
95 |
96 | } else {
97 |
98 | sign_out();
99 |
100 | }
101 |
102 | auth_loop();
103 |
104 | });
105 |
106 |
107 |
108 | }, delay);
109 |
110 |
111 | }
112 |
113 |
114 |
115 | function do_auth() {
116 |
117 | var form_data = $('#auth_form').serializeArray();
118 |
119 | $.post('api2/do_auth', form_data, function(data) {
120 |
121 | if (data.auth_status) {
122 |
123 | update_cookie(data.auth_cookie);
124 | location.reload();
125 |
126 | } else {
127 |
128 | remove_cookie();
129 |
130 | $('.auth-form-username').toggleClass('is-invalid', true);
131 | $('.auth-form-password').toggleClass('is-invalid', true);
132 | $('.auth-form-otp').toggleClass('is-invalid', true);
133 |
134 | }
135 |
136 |
137 |
138 | });
139 |
140 |
141 |
142 | return false;
143 |
144 | }
145 |
146 |
147 |
148 | function save_records() {
149 |
150 | var zone = $('#zones :selected').val();
151 | if (zone && zone != 0) {
152 |
153 | var form_data = $('#records_form').serializeArray();
154 |
155 | var all_valid = true;
156 |
157 | $('#records_form .record-data').each(function( index ) {
158 |
159 | if (this.value == '') {
160 | all_valid = false;
161 | $(this).toggleClass('is-invalid', true);
162 | } else {
163 | $(this).removeClass('is-invalid');
164 | }
165 |
166 | });
167 |
168 |
169 | if (all_valid) {
170 |
171 | $.post('api2/save_records', form_data, function(data) {
172 | get_records(zone);
173 | toastr.success('Records has been saved!');
174 | });
175 |
176 | }
177 |
178 |
179 | }
180 |
181 | }
182 |
183 |
184 | function remove_record(zone, type, name) {
185 |
186 | if (confirm('Proceed with the "' + type + '" type for "' + name + '" record removal?')) {
187 | $.post('api2/remove_record', { zone: zone, type: type, name: name }, function(data) {
188 | get_records(zone);
189 | });
190 | }
191 |
192 | }
193 |
194 |
195 |
196 | function remove_row(row_id) {
197 | $('#row_' + row_id).remove();
198 | }
199 |
200 |
201 | function remove_zone() {
202 | var zone = $('#zones :selected').val();
203 | if (zone && zone != 0) {
204 | if (confirm('Proceed with the "' + zone + '" zone removal?')) {
205 | $.post('api2/remove_zone/' + zone, function(data) {
206 | clear_records();
207 | refresh_zones();
208 | });
209 | }
210 | }
211 | }
212 |
213 |
214 |
215 | function refresh_records() {
216 | var zone = $('#zones :selected').val();
217 | if (zone && zone != 0) {
218 | get_records(zone);
219 | }
220 | refresh_zones(zone);
221 |
222 | toastr.info('Records has been re-loaded...');
223 |
224 | }
225 |
226 |
227 |
228 | function clear_records() {
229 | $('#records').empty();
230 | }
231 |
232 |
233 |
234 | function add_zone(params) {
235 | $.post('api2/add_zone', params, function (data) {
236 | clear_records();
237 | refresh_zones();
238 | toastr.success('The new zone [' + params['zone'] + '] has been created!');
239 | });
240 | }
241 |
242 |
243 | function get_records(zone) {
244 |
245 | $.post('api2/get_records/' + zone, function(data) {
246 |
247 |
248 | clear_records();
249 |
250 |
251 | for (let key in data) {
252 |
253 | // console.log(data[key]);
254 |
255 | var type = data[key].type;
256 | var name = data[key].name;
257 | var records = data[key].records;
258 | var ttl = data[key].ttl;
259 |
260 | // console.log(records);
261 |
262 |
263 | records.forEach(function(value, index, array) {
264 |
265 | var content = value.content;
266 |
267 | var record_tpl = ``;
274 |
275 | $('#records').append(record_tpl);
276 |
277 | row_id++;
278 |
279 | });
280 |
281 |
282 |
283 |
284 | }
285 |
286 |
287 | var add_record_tpl = `
288 |
289 |
290 |
291 |
`;
292 |
293 | $('#records').append(add_record_tpl);
294 |
295 |
296 | });
297 |
298 |
299 | }
300 |
301 | function set_type(row_id, type, zone) {
302 |
303 | var type_val = '';
304 | var name_attr = zone;
305 | var content_attr = '';
306 |
307 |
308 | if (type != '') {
309 |
310 | type_val = type;
311 |
312 | if (type == 'TXT') {
313 | content_attr = '"some text value surrounded by quotes"';
314 | } else if (type == 'A') {
315 | content_attr = '127.1.2.3';
316 | } else if (type == 'CNAME') {
317 | content_attr = 'domain01.' + zone;
318 | } else if (type == 'MX') {
319 | content_attr = '10 ' + zone;
320 | } else if (type == 'SRV') {
321 | name_attr = '_sip._udp.sip.' + zone;
322 | content_attr = 'prio[10] weight[1] port[5060] sip.' + zone;
323 | }
324 |
325 | }
326 |
327 |
328 | $('#type_field_' + row_id).val(type_val);
329 | $('#name_field_' + row_id).attr('placeholder', name_attr);
330 | $('#content_field_' + row_id).attr('placeholder', content_attr);
331 |
332 |
333 | }
334 |
335 |
336 |
337 | function add_record(zone) {
338 |
339 | var dh = $(document).height();
340 |
341 |
342 | var new_record_tpl = `
343 |
344 |
352 |
353 |
354 |
355 |
356 |
357 |
358 |
`;
359 |
360 |
361 | $(new_record_tpl).insertBefore( $('#add_record_btn_div') );
362 | row_id++;
363 |
364 |
365 |
366 | var diff = $(document).height() - dh;
367 | window.scrollBy(0, diff);
368 |
369 |
370 |
371 | }
372 |
373 |
374 |
375 | function refresh_zones(default_zone = null) {
376 |
377 | $.post('api2/get_zones', function(data) {
378 |
379 | var list = $('#zones').empty();
380 | list.append($('