mirror of
				https://github.com/mail-in-a-box/mailinabox.git
				synced 2025-10-22 17:30:55 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			472 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			472 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <!DOCTYPE html>
 | |
| <html lang="en">
 | |
|     <head>
 | |
|         <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
 | |
|         <meta charset="utf-8">
 | |
|         <meta name="viewport" content="width=device-width, initial-scale=1">
 | |
| 
 | |
|         <title>{{hostname}} - Mail-in-a-Box Control Panel</title>
 | |
| 
 | |
|         <meta name="robots" content="noindex, nofollow">
 | |
| 
 | |
|         <link rel="stylesheet" href="/admin/assets/bootstrap/css/bootstrap.min.css">
 | |
|         <style>
 | |
| 	    body {
 | |
| 		overflow-y: scroll;
 | |
|                 padding-bottom: 20px;
 | |
|             }
 | |
| 
 | |
|             p {
 | |
|               margin-bottom: 1.25em;
 | |
|             }
 | |
| 
 | |
|             h1, h2, h3, h4 {
 | |
|               font-family: sans-serif;
 | |
|               font-weight: bold;
 | |
|             }
 | |
| 
 | |
|             h2 {
 | |
|               margin: 1em 0;
 | |
|             }
 | |
| 
 | |
|             h3 {
 | |
|               font-size: 130%;
 | |
|               border-bottom: 1px solid black;
 | |
|               padding-bottom: 3px;
 | |
|               margin-bottom: 13px;
 | |
|               margin-top: 30px;
 | |
|             }
 | |
|               .panel-heading h3 {
 | |
|                 border: none;
 | |
|                 padding: 0;
 | |
|                 margin: 0;
 | |
|               }
 | |
| 
 | |
|             h4 {
 | |
|               font-size: 110%;
 | |
|               margin-bottom: 13px;
 | |
|               margin-top: 18px;
 | |
|             }
 | |
|               h4:first-child {
 | |
|                 margin-top: 6px;
 | |
|               }
 | |
| 
 | |
|             .admin_panel {
 | |
|               display: none;
 | |
|             }
 | |
| 
 | |
|             table.table {
 | |
|               margin: 1.5em 0;
 | |
|             }
 | |
| 
 | |
| 	    ol li {
 | |
| 	       margin-bottom: 1em;
 | |
| 	    }
 | |
|           
 | |
|             .if-logged-in { display: none; }
 | |
|             .if-logged-in-admin { display: none; }
 | |
| 
 | |
|       /* The below only gets used if it is supported */
 | |
|       @media (prefers-color-scheme: dark) {
 | |
|         /* Invert invert lightness but not hue */
 | |
|         html {
 | |
|           filter: invert(100%) hue-rotate(180deg);
 | |
|         }
 | |
| 
 | |
|         /* Set explicit background color (necessary for Firefox) */
 | |
|         html {
 | |
|           background-color: #111;
 | |
|         }
 | |
|           
 | |
|         /* Override Boostrap theme here to give more contrast. The black turns to white by the filter. */
 | |
|         .form-control {
 | |
|           color: black !important;
 | |
|         }          
 | |
| 
 | |
|         /* Revert the invert for the navbar */
 | |
|         button, div.navbar {
 | |
|           filter: invert(100%) hue-rotate(180deg);
 | |
|         }
 | |
|           
 | |
|         /* Revert the revert for the dropdowns */
 | |
|         ul.dropdown-menu {
 | |
|           filter: invert(100%) hue-rotate(180deg);
 | |
|         }
 | |
|       }
 | |
|         </style>
 | |
|         <link rel="stylesheet" href="/admin/assets/bootstrap/css/bootstrap-theme.min.css">
 | |
|     </head>
 | |
|     <body>
 | |
| 
 | |
|     <!--[if lt IE 8]><p>Internet Explorer version 8 or any modern web browser is required to use this website, sorry.<![endif]-->
 | |
|     <!--[if gt IE 7]><!-->
 | |
| 
 | |
|     <div class="navbar navbar-inverse navbar-static-top" role="navigation">
 | |
|       <div class="container">
 | |
|         <div class="navbar-header">
 | |
|           <button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target=".navbar-collapse">
 | |
|             <span class="sr-only">Toggle navigation</span>
 | |
|             <span class="icon-bar"></span>
 | |
|             <span class="icon-bar"></span>
 | |
|             <span class="icon-bar"></span>
 | |
|           </button>
 | |
|           <a class="navbar-brand" href="#">{{hostname}}</a>
 | |
|         </div>
 | |
|         <div class="navbar-collapse collapse">
 | |
|           <ul class="nav navbar-nav">
 | |
|             <li class="dropdown if-logged-in-admin">
 | |
|               <a href="#" class="dropdown-toggle" data-toggle="dropdown">System <b class="caret"></b></a>
 | |
|               <ul class="dropdown-menu">
 | |
|                 <li><a href="#system_status" onclick="return show_panel(this);">Status Checks</a></li>
 | |
|                 <li><a href="#tls" onclick="return show_panel(this);">TLS (SSL) Certificates</a></li>
 | |
|                 <li><a href="#system_backup" onclick="return show_panel(this);">Backup Status</a></li>
 | |
|                 <li class="divider"></li>
 | |
|                 <li class="dropdown-header">Advanced Pages</li>
 | |
|                 <li><a href="#custom_dns" onclick="return show_panel(this);">Custom DNS</a></li>
 | |
|                 <li><a href="#external_dns" onclick="return show_panel(this);">External DNS</a></li>
 | |
|                 <li><a href="#munin" onclick="return show_panel(this);">Munin Monitoring</a></li>
 | |
|               </ul>
 | |
|             </li>
 | |
|             <li><a href="#mail-guide" onclick="return show_panel(this);" class="if-logged-in-not-admin">Mail</a></li>
 | |
|             <li class="dropdown if-logged-in-admin">
 | |
|               <a href="#" class="dropdown-toggle" data-toggle="dropdown">Mail & Users <b class="caret"></b></a>
 | |
|               <ul class="dropdown-menu">
 | |
|                 <li><a href="#mail-guide" onclick="return show_panel(this);">Instructions</a></li>
 | |
|                 <li><a href="#users" onclick="return show_panel(this);">Users</a></li>
 | |
|                 <li><a href="#aliases" onclick="return show_panel(this);">Aliases</a></li>
 | |
|                 <li class="divider"></li>
 | |
|                 <li class="dropdown-header">Your Account</li>
 | |
|                 <li><a href="#mfa" onclick="return show_panel(this);">Two-Factor Authentication</a></li>
 | |
|               </ul>
 | |
|             </li>
 | |
|             <li><a href="#sync_guide" onclick="return show_panel(this);" class="if-logged-in">Contacts/Calendar</a></li>
 | |
|             <li><a href="#web" onclick="return show_panel(this);" class="if-logged-in-admin">Web</a></li>
 | |
|           </ul>
 | |
|           <ul class="nav navbar-nav navbar-right">
 | |
|             <li class="if-logged-in"><a href="#" onclick="do_logout(); return false;" style="color: white">Log out</a></li>
 | |
|           </ul>
 | |
|         </div><!--/.navbar-collapse -->
 | |
|       </div>
 | |
|     </div>
 | |
| 
 | |
|     <div class="container">
 | |
|       <div id="panel_welcome" class="admin_panel">
 | |
|       {% include "welcome.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_system_status" class="admin_panel">
 | |
|       {% include "system-status.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_system_backup" class="admin_panel">
 | |
|       {% include "system-backup.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_external_dns" class="admin_panel">
 | |
|       {% include "external-dns.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_custom_dns" class="admin_panel">
 | |
|       {% include "custom-dns.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_mfa" class="admin_panel">
 | |
|       {% include "mfa.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_login" class="admin_panel">
 | |
|       {% include "login.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_mail-guide" class="admin_panel">
 | |
|       {% include "mail-guide.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_users" class="admin_panel">
 | |
|       {% include "users.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_aliases" class="admin_panel">
 | |
|       {% include "aliases.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_sync_guide" class="admin_panel">
 | |
|       {% include "sync-guide.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_web" class="admin_panel">
 | |
|       {% include "web.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_tls" class="admin_panel">
 | |
|       {% include "ssl.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <div id="panel_munin" class="admin_panel">
 | |
|       {% include "munin.html" %}
 | |
|       </div>
 | |
| 
 | |
|       <hr>
 | |
| 
 | |
|       <footer>
 | |
|         <p>This is a <a href="https://mailinabox.email">Mail-in-a-Box</a>.</p>
 | |
|       </footer>
 | |
|     </div> <!-- /container -->
 | |
| 
 | |
|         <div id="ajax_loading_indicator" style="display: none; position: fixed; left: 0; top: 0; width: 100%; height: 100%; z-index: 100000; text-align: center; background-color: rgba(255,255,255,.75)">
 | |
|           <div style="margin: 20% auto">
 | |
|             <div><span class="fa fa-spinner fa-pulse"></span></div>
 | |
|             <div>Loading...</div>
 | |
|           </div>
 | |
|         </div>
 | |
| 
 | |
|         <div id="global_modal" class="modal fade" tabindex="-1" role="dialog" aria-labelledby="errorModalTitle" aria-hidden="true">
 | |
|           <div class="modal-dialog modal-sm">
 | |
|             <div class="modal-content">
 | |
|               <div class="modal-header">
 | |
|                 <button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button>
 | |
|                 <h4 class="modal-title" id="errorModalTitle"> </h4>
 | |
|               </div>
 | |
|               <div class="modal-body">
 | |
|                 <p> </p>
 | |
|               </div>
 | |
|               <div class="modal-footer">
 | |
|                 <button type="button" class="btn btn-default" data-dismiss="modal">OK</button>
 | |
|                 <button type="button" class="btn btn-danger" data-dismiss="modal">Yes</button>
 | |
|               </div>
 | |
|             </div>
 | |
|           </div>
 | |
|         </div>
 | |
| 
 | |
|         <script src="/admin/assets/jquery.min.js"></script>
 | |
|         <script src="/admin/assets/bootstrap/js/bootstrap.min.js"></script>
 | |
| 
 | |
|         <script>
 | |
| var global_modal_state = null;
 | |
| var global_modal_funcs = null;
 | |
| 
 | |
| $(function() {
 | |
|   $('#global_modal').on('shown.bs.modal', function (e) {
 | |
|     // set focus to first input in the global modal's body
 | |
|     var input = $('#global_modal .modal-body input');
 | |
|     if (input.length > 0) $(input[0]).focus();
 | |
|   })
 | |
|   $('#global_modal .btn-danger').click(function() {
 | |
|     // Don't take action now. Wait for the modal to be totally hidden
 | |
|     // so that we don't attempt to show another modal while this one
 | |
|     // is closing.
 | |
|     global_modal_state = 0; // OK
 | |
|   })
 | |
|   $('#global_modal .btn-default').click(function() {
 | |
|     global_modal_state = 1; // Cancel
 | |
|   })
 | |
|   $('#global_modal').on('hidden.bs.modal', function (e) {
 | |
|     // do the cancel function
 | |
|     if (global_modal_state == null) global_modal_state = 1; // cancel if the user hit ESC or clicked outside of the modal
 | |
|     if (global_modal_funcs && global_modal_funcs[global_modal_state])
 | |
|       global_modal_funcs[global_modal_state]();
 | |
|   })
 | |
| })
 | |
| 
 | |
| function show_modal_error(title, message, callback) {
 | |
|   $('#global_modal h4').text(title);
 | |
|   $('#global_modal .modal-body').html("<p/>");
 | |
|   if (typeof question == 'string') {
 | |
|     $('#global_modal p').text(message);
 | |
|     $('#global_modal .modal-dialog').addClass("modal-sm");
 | |
|   } else {
 | |
|     $('#global_modal p').html("").append(message);
 | |
|     $('#global_modal .modal-dialog').removeClass("modal-sm");
 | |
|   }
 | |
|   $('#global_modal .btn-default').show().text("OK");
 | |
|   $('#global_modal .btn-danger').hide();
 | |
|   global_modal_funcs = [callback, callback];
 | |
|   global_modal_state = null;
 | |
|   $('#global_modal').modal({});
 | |
|   return false; // handy when called from onclick
 | |
| }
 | |
| 
 | |
| function show_modal_confirm(title, question, verb, yes_callback, cancel_callback) {
 | |
|   $('#global_modal h4').text(title);
 | |
|   if (typeof question == 'string') {
 | |
|     $('#global_modal .modal-dialog').addClass("modal-sm");
 | |
|     $('#global_modal .modal-body').html("<p/>");
 | |
|     $('#global_modal p').text(question);
 | |
|   } else {
 | |
|     $('#global_modal .modal-dialog').removeClass("modal-sm");
 | |
|     $('#global_modal .modal-body').html("").append(question);
 | |
|   }
 | |
|   if (typeof verb == 'string') {
 | |
|     $('#global_modal .btn-default').show().text("Cancel");
 | |
|     $('#global_modal .btn-danger').show().text(verb);
 | |
|   } else {
 | |
|     $('#global_modal .btn-default').show().text(verb[1]);
 | |
|     $('#global_modal .btn-danger').show().text(verb[0]);
 | |
|   }
 | |
|   global_modal_funcs = [yes_callback, cancel_callback];
 | |
|   global_modal_state = null;
 | |
|   $('#global_modal').modal({});
 | |
|   return false; // handy when called from onclick
 | |
| }
 | |
| 
 | |
| var ajax_num_executing_requests = 0;
 | |
| function ajax_with_indicator(options) {
 | |
|   setTimeout("if (ajax_num_executing_requests > 0) $('#ajax_loading_indicator').fadeIn()", 100);
 | |
|   function hide_loading_indicator() {
 | |
|     ajax_num_executing_requests--;
 | |
|     if (ajax_num_executing_requests == 0)
 | |
|       $('#ajax_loading_indicator').stop(true).hide(); // stop() prevents an ongoing fade from causing the thing to be shown again after this call
 | |
|   }
 | |
|   var old_success = options.success;
 | |
|   var old_error = options.error;
 | |
|   options.success = function(data) {
 | |
|     hide_loading_indicator();
 | |
|     if (data.status == "error")
 | |
|       show_modal_error("Error", data.message);
 | |
|     else if (old_success)
 | |
|       old_success(data);
 | |
|   };
 | |
|   options.error = function(jqxhr) {
 | |
|     hide_loading_indicator();
 | |
|     if (!old_error)
 | |
|       show_modal_error("Error", "Something went wrong, sorry.")
 | |
|     else
 | |
|       old_error(jqxhr.responseText, jqxhr);
 | |
|   };
 | |
|   ajax_num_executing_requests++;
 | |
|   $.ajax(options);
 | |
|   return false; // handy when called from onclick
 | |
| }
 | |
| 
 | |
| var api_credentials = null;
 | |
| function api(url, method, data, callback, callback_error, headers) {
 | |
|   // from http://www.webtoolkit.info/javascript-base64.html
 | |
|   function base64encode(input) {
 | |
|     _keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
 | |
|     var output = "";
 | |
|     var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
 | |
|     var i = 0;
 | |
|     while (i < input.length) {
 | |
|         chr1 = input.charCodeAt(i++);
 | |
|         chr2 = input.charCodeAt(i++);
 | |
|         chr3 = input.charCodeAt(i++);
 | |
|         enc1 = chr1 >> 2;
 | |
|         enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
 | |
|         enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
 | |
|         enc4 = chr3 & 63;
 | |
|         if (isNaN(chr2)) {
 | |
|             enc3 = enc4 = 64;
 | |
|         } else if (isNaN(chr3)) {
 | |
|             enc4 = 64;
 | |
|         }
 | |
|         output = output +
 | |
|         _keyStr.charAt(enc1) + _keyStr.charAt(enc2) +
 | |
|         _keyStr.charAt(enc3) + _keyStr.charAt(enc4);
 | |
|     }
 | |
| 
 | |
|     return output;
 | |
|   }
 | |
| 
 | |
|   function default_error(text, xhr) {
 | |
|     if (xhr.status != 403) // else handled below
 | |
|       show_modal_error("Error", "Something went wrong, sorry.")
 | |
|   }
 | |
| 
 | |
|   ajax_with_indicator({
 | |
|     url: "/admin" + url,
 | |
|     method: method,
 | |
|     cache: false,
 | |
|     data: data,
 | |
|     headers: headers,
 | |
|     // the custom DNS api sends raw POST/PUT bodies --- prevent URL-encoding
 | |
|     processData: typeof data != "string",
 | |
|     mimeType: typeof data == "string" ? "text/plain; charset=ascii" : null,
 | |
| 
 | |
|     beforeSend: function(xhr) {
 | |
|       // We don't store user credentials in a cookie to avoid the hassle of CSRF
 | |
|       // attacks. The Authorization header only gets set in our AJAX calls triggered
 | |
|       // by user actions.
 | |
|       if (api_credentials)
 | |
|         xhr.setRequestHeader(
 | |
|           'Authorization',
 | |
|           'Basic ' + base64encode(api_credentials.username + ':' + api_credentials.session_key));
 | |
|     },
 | |
|     success: callback,
 | |
|     error: callback_error || default_error,
 | |
|     statusCode: {
 | |
|       403: function(xhr) {
 | |
|         // Credentials are no longer valid. Try to login again.
 | |
|         var p = current_panel;
 | |
|         show_panel('login');
 | |
|         switch_back_to_panel = p;
 | |
|       }
 | |
|     }
 | |
|   })
 | |
| }
 | |
| 
 | |
| var current_panel = null;
 | |
| var switch_back_to_panel = null;
 | |
| 
 | |
| function do_logout() {
 | |
|   // Clear the session from the backend.
 | |
|   api("/logout", "POST");
 | |
| 
 | |
|   // Forget the token.
 | |
|   api_credentials = null;
 | |
|   if (typeof localStorage != 'undefined')
 | |
|     localStorage.removeItem("miab-cp-credentials");
 | |
|   if (typeof sessionStorage != 'undefined')
 | |
|     sessionStorage.removeItem("miab-cp-credentials");
 | |
| 
 | |
|   // Return to the start.
 | |
|   show_panel('login');
 | |
| 
 | |
|   // Reset menus.
 | |
|   show_hide_menus();
 | |
| }
 | |
| 
 | |
| function show_panel(panelid) {
 | |
|   if (panelid.getAttribute)
 | |
|     // we might be passed an HTMLElement <a>.
 | |
|     panelid = panelid.getAttribute('href').substring(1);
 | |
| 
 | |
|   $('.admin_panel').hide();
 | |
|   $('#panel_' + panelid).show();
 | |
|   if (typeof localStorage != 'undefined')
 | |
|     localStorage.setItem("miab-cp-lastpanel", panelid);
 | |
|   if (window["show_" + panelid])
 | |
|     window["show_" + panelid]();
 | |
| 
 | |
|   current_panel = panelid;
 | |
|   switch_back_to_panel = null;
 | |
| 
 | |
|   return false; // when called from onclick, cancel navigation
 | |
| }
 | |
| 
 | |
| $(function() {
 | |
|   // Recall saved user credentials.
 | |
|   try {
 | |
|     if (typeof sessionStorage != 'undefined' && sessionStorage.getItem("miab-cp-credentials"))
 | |
|       api_credentials = JSON.parse(sessionStorage.getItem("miab-cp-credentials"));
 | |
|     else if (typeof localStorage != 'undefined' && localStorage.getItem("miab-cp-credentials"))
 | |
|       api_credentials = JSON.parse(localStorage.getItem("miab-cp-credentials"));
 | |
|   } catch (_) {
 | |
|   }
 | |
| 
 | |
|   // Toggle menu state.
 | |
|   show_hide_menus();
 | |
| 
 | |
|   // Recall what the user was last looking at.
 | |
|   if (api_credentials != null && typeof localStorage != 'undefined' && localStorage.getItem("miab-cp-lastpanel")) {
 | |
|     show_panel(localStorage.getItem("miab-cp-lastpanel"));
 | |
|   } else if (api_credentials != null) {
 | |
|     show_panel('welcome');
 | |
|   } else {
 | |
|     show_panel('login');
 | |
|   }
 | |
| })
 | |
| 
 | |
|         </script>
 | |
|     </body>
 | |
| </html>
 |