357 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			357 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| @extends('layouts.admin')
 | |
| 
 | |
| @section('title')
 | |
|     {{ $node->name }}: Allocations
 | |
| @endsection
 | |
| 
 | |
| @section('content-header')
 | |
|     <h1>{{ $node->name }}<small>Control allocations available for servers on this node.</small></h1>
 | |
|     <ol class="breadcrumb">
 | |
|         <li><a href="{{ route('admin.index') }}">Admin</a></li>
 | |
|         <li><a href="{{ route('admin.nodes') }}">Nodes</a></li>
 | |
|         <li><a href="{{ route('admin.nodes.view', $node->id) }}">{{ $node->name }}</a></li>
 | |
|         <li class="active">Allocations</li>
 | |
|     </ol>
 | |
| @endsection
 | |
| 
 | |
| @section('content')
 | |
| <div class="row">
 | |
|     <div class="col-xs-12">
 | |
|         <div class="nav-tabs-custom nav-tabs-floating">
 | |
|             <ul class="nav nav-tabs">
 | |
|                 <li><a href="{{ route('admin.nodes.view', $node->id) }}">About</a></li>
 | |
|                 <li><a href="{{ route('admin.nodes.view.settings', $node->id) }}">Settings</a></li>
 | |
|                 <li><a href="{{ route('admin.nodes.view.configuration', $node->id) }}">Configuration</a></li>
 | |
|                 <li class="active"><a href="{{ route('admin.nodes.view.allocation', $node->id) }}">Allocation</a></li>
 | |
|                 <li><a href="{{ route('admin.nodes.view.servers', $node->id) }}">Servers</a></li>
 | |
|             </ul>
 | |
|         </div>
 | |
|     </div>
 | |
| </div>
 | |
| <div class="row">
 | |
|     <div class="col-sm-8">
 | |
|         <div class="box box-primary">
 | |
|             <div class="box-header with-border">
 | |
|                 <h3 class="box-title">Existing Allocations</h3>
 | |
|             </div>
 | |
|             <div class="box-body table-responsive no-padding" style="overflow-x: visible">
 | |
|                 <table class="table table-hover" style="margin-bottom:0;">
 | |
|                     <tr>
 | |
|                         <th>
 | |
|                             <input type="checkbox" class="select-all-files hidden-xs" data-action="selectAll">
 | |
|                         </th>
 | |
|                         <th>IP Address <i class="fa fa-fw fa-minus-square" style="font-weight:normal;color:#d9534f;cursor:pointer;" data-toggle="modal" data-target="#allocationModal"></i></th>
 | |
|                         <th>IP Alias</th>
 | |
|                         <th>Port</th>
 | |
|                         <th>Assigned To</th>
 | |
|                         <th>
 | |
|                             <div class="btn-group hidden-xs">
 | |
|                                 <button type="button" id="mass_actions" class="btn btn-sm btn-default dropdown-toggle disabled"
 | |
|                                         data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">Mass Actions <span class="caret"></span>
 | |
|                                 </button>
 | |
|                                 <ul class="dropdown-menu dropdown-massactions">
 | |
|                                     <li><a href="#" id="selective-deletion" data-action="selective-deletion">Delete <i class="fa fa-fw fa-trash-o"></i></a></li>
 | |
|                                 </ul>
 | |
|                             </div>
 | |
|                         </th>
 | |
|                     </tr>
 | |
|                     @foreach($node->allocations as $allocation)
 | |
|                         <tr>
 | |
|                             <td class="middle min-size" data-identifier="type">
 | |
|                                 @if(is_null($allocation->server_id))
 | |
|                                 <input type="checkbox" class="select-file hidden-xs" data-action="addSelection">
 | |
|                                 @else
 | |
|                                 <input disabled="disabled" type="checkbox" class="select-file hidden-xs" data-action="addSelection">
 | |
|                                 @endif
 | |
|                             </td>
 | |
|                             <td class="col-sm-3 middle" data-identifier="ip">{{ $allocation->ip }}</td>
 | |
|                             <td class="col-sm-3 middle">
 | |
|                                 <input class="form-control input-sm" type="text" value="{{ $allocation->ip_alias }}" data-action="set-alias" data-id="{{ $allocation->id }}" placeholder="none" />
 | |
|                                 <span class="input-loader"><i class="fa fa-refresh fa-spin fa-fw"></i></span>
 | |
|                             </td>
 | |
|                             <td class="col-sm-2 middle" data-identifier="port">{{ $allocation->port }}</td>
 | |
|                             <td class="col-sm-3 middle">
 | |
|                                 @if(! is_null($allocation->server))
 | |
|                                     <a href="{{ route('admin.servers.view', $allocation->server_id) }}">{{ $allocation->server->name }}</a>
 | |
|                                 @endif
 | |
|                             </td>
 | |
|                             <td class="col-sm-1 middle">
 | |
|                                 @if(is_null($allocation->server_id))
 | |
|                                     <button data-action="deallocate" data-id="{{ $allocation->id }}" class="btn btn-sm btn-danger"><i class="fa fa-trash-o"></i></button>
 | |
|                                 @endif
 | |
|                             </td>
 | |
|                         </tr>
 | |
|                     @endforeach
 | |
|                 </table>
 | |
|             </div>
 | |
|             @if($node->allocations->hasPages())
 | |
|                 <div class="box-footer text-center">
 | |
|                     {{ $node->allocations->render() }}
 | |
|                 </div>
 | |
|             @endif
 | |
|         </div>
 | |
|     </div>
 | |
|     <div class="col-sm-4">
 | |
|         <form action="{{ route('admin.nodes.view.allocation', $node->id) }}" method="POST">
 | |
|             <div class="box box-success">
 | |
|                 <div class="box-header with-border">
 | |
|                     <h3 class="box-title">Assign New Allocations</h3>
 | |
|                 </div>
 | |
|                 <div class="box-body">
 | |
|                     <div class="form-group">
 | |
|                         <label for="pAllocationIP" class="control-label">IP Address</label>
 | |
|                         <div>
 | |
|                             <select class="form-control" name="allocation_ip" id="pAllocationIP" multiple>
 | |
|                                 @foreach($allocations as $allocation)
 | |
|                                     <option value="{{ $allocation->ip }}">{{ $allocation->ip }}</option>
 | |
|                                 @endforeach
 | |
|                             </select>
 | |
|                             <p class="text-muted small">Enter an IP address to assign ports to here.</p>
 | |
|                         </div>
 | |
|                     </div>
 | |
|                     <div class="form-group">
 | |
|                         <label for="pAllocationIP" class="control-label">IP Alias</label>
 | |
|                         <div>
 | |
|                             <input type="text" id="pAllocationAlias" class="form-control" name="allocation_alias" placeholder="alias" />
 | |
|                             <p class="text-muted small">If you would like to assign a default alias to these allocations enter it here.</p>
 | |
|                         </div>
 | |
|                     </div>
 | |
|                     <div class="form-group">
 | |
|                         <label for="pAllocationPorts" class="control-label">Ports</label>
 | |
|                         <div>
 | |
|                             <select class="form-control" name="allocation_ports[]" id="pAllocationPorts" multiple></select>
 | |
|                             <p class="text-muted small">Enter individual ports or port ranges here separated by commas or spaces.</p>
 | |
|                         </div>
 | |
|                     </div>
 | |
|                 </div>
 | |
|                 <div class="box-footer">
 | |
|                     {!! csrf_field() !!}
 | |
|                     <button type="submit" class="btn btn-success btn-sm pull-right">Submit</button>
 | |
|                 </div>
 | |
|             </div>
 | |
|         </form>
 | |
|     </div>
 | |
| </div>
 | |
| <div class="modal fade" id="allocationModal" tabindex="-1" role="dialog">
 | |
|     <div class="modal-dialog" role="document">
 | |
|         <div class="modal-content">
 | |
|             <div class="modal-header">
 | |
|                 <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">×</span></button>
 | |
|                 <h4 class="modal-title">Delete Allocations for IP Block</h4>
 | |
|             </div>
 | |
|             <form action="{{ route('admin.nodes.view.allocation.removeBlock', $node->id) }}" method="POST">
 | |
|                 <div class="modal-body">
 | |
|                     <div class="row">
 | |
|                         <div class="col-md-12">
 | |
|                             <select class="form-control" name="ip">
 | |
|                                 @foreach($allocations as $allocation)
 | |
|                                     <option value="{{ $allocation->ip }}">{{ $allocation->ip }}</option>
 | |
|                                 @endforeach
 | |
|                             </select>
 | |
|                         </div>
 | |
|                     </div>
 | |
|                 </div>
 | |
|                 <div class="modal-footer">
 | |
|                     {{{ csrf_field() }}}
 | |
|                     <button type="button" class="btn btn-default" data-dismiss="modal">Close</button>
 | |
|                     <button type="submit" class="btn btn-danger">Delete Allocations</button>
 | |
|                 </div>
 | |
|             </form>
 | |
|         </div>
 | |
|     </div>
 | |
| </div>
 | |
| @endsection
 | |
| 
 | |
| @section('footer-scripts')
 | |
|     @parent
 | |
|     <script>
 | |
|     $('[data-action="addSelection"]').on('click', function () {
 | |
|         updateMassActions();
 | |
|     });
 | |
| 
 | |
|     $('[data-action="selectAll"]').on('click', function () {
 | |
|         $('input.select-file').not(':disabled').prop('checked', function (i, val) {
 | |
|             return !val;
 | |
|         });
 | |
| 
 | |
|         updateMassActions();
 | |
|     });
 | |
| 
 | |
|     $('[data-action="selective-deletion"]').on('mousedown', function () {
 | |
|         deleteSelected();
 | |
|     });
 | |
| 
 | |
|     $('#pAllocationIP').select2({
 | |
|         tags: true,
 | |
|         maximumSelectionLength: 1,
 | |
|         selectOnClose: true,
 | |
|         tokenSeparators: [',', ' '],
 | |
|     });
 | |
| 
 | |
|     $('#pAllocationPorts').select2({
 | |
|         tags: true,
 | |
|         selectOnClose: true,
 | |
|         tokenSeparators: [',', ' '],
 | |
|     });
 | |
| 
 | |
|     $('button[data-action="deallocate"]').click(function (event) {
 | |
|         event.preventDefault();
 | |
|         var element = $(this);
 | |
|         var allocation = $(this).data('id');
 | |
|         swal({
 | |
|             title: '',
 | |
|             text: 'Are you sure you want to delete this allocation?',
 | |
|             type: 'warning',
 | |
|             showCancelButton: true,
 | |
|             allowOutsideClick: true,
 | |
|             closeOnConfirm: false,
 | |
|             confirmButtonText: 'Delete',
 | |
|             confirmButtonColor: '#d9534f',
 | |
|             showLoaderOnConfirm: true
 | |
|         }, function () {
 | |
|             $.ajax({
 | |
|                 method: 'DELETE',
 | |
|                 url: '/admin/nodes/view/' + {{ $node->id }} + '/allocation/remove/' + allocation,
 | |
|                 headers: { 'X-CSRF-TOKEN': $('meta[name="_token"]').attr('content') },
 | |
|             }).done(function (data) {
 | |
|                 element.parent().parent().addClass('warning').delay(100).fadeOut();
 | |
|                 swal({ type: 'success', title: 'Port Deleted!' });
 | |
|             }).fail(function (jqXHR) {
 | |
|                 console.error(jqXHR);
 | |
|                 swal({
 | |
|                     title: 'Whoops!',
 | |
|                     text: jqXHR.responseJSON.error,
 | |
|                     type: 'error'
 | |
|                 });
 | |
|             });
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     var typingTimer;
 | |
|     $('input[data-action="set-alias"]').keyup(function () {
 | |
|         clearTimeout(typingTimer);
 | |
|         $(this).parent().removeClass('has-error has-success');
 | |
|         typingTimer = setTimeout(sendAlias, 250, $(this));
 | |
|     });
 | |
| 
 | |
|     var fadeTimers = [];
 | |
|     function sendAlias(element) {
 | |
|         element.parent().find('.input-loader').show();
 | |
|         clearTimeout(fadeTimers[element.data('id')]);
 | |
|         $.ajax({
 | |
|             method: 'POST',
 | |
|             url: '/admin/nodes/view/' + {{ $node->id }} + '/allocation/alias',
 | |
|             headers: { 'X-CSRF-TOKEN': $('meta[name="_token"]').attr('content') },
 | |
|             data: {
 | |
|                 alias: element.val(),
 | |
|                 allocation_id: element.data('id'),
 | |
|             }
 | |
|         }).done(function () {
 | |
|             element.parent().addClass('has-success');
 | |
|         }).fail(function (jqXHR) {
 | |
|             console.error(jqXHR);
 | |
|             element.parent().addClass('has-error');
 | |
|         }).always(function () {
 | |
|             element.parent().find('.input-loader').hide();
 | |
|             fadeTimers[element.data('id')] = setTimeout(clearHighlight, 2500, element);
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     function clearHighlight(element) {
 | |
|         element.parent().removeClass('has-error has-success');
 | |
|     }
 | |
| 
 | |
|     function updateMassActions() {
 | |
|         if ($('input.select-file:checked').length > 0) {
 | |
|             $('#mass_actions').removeClass('disabled');
 | |
|         } else {
 | |
|             $('#mass_actions').addClass('disabled');
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     function deleteSelected() {
 | |
|         var selectedIds = [];
 | |
|         var selectedItems = [];
 | |
|         var selectedItemsElements = [];
 | |
| 
 | |
|         $('input.select-file:checked').each(function () {
 | |
|             var $parent = $($(this).closest('tr'));
 | |
|             var id = $parent.find('[data-action="deallocate"]').data('id');
 | |
|             var $ip = $parent.find('td[data-identifier="ip"]');
 | |
|             var $port = $parent.find('td[data-identifier="port"]');
 | |
|             var block = `${$ip.text()}:${$port.text()}`;
 | |
| 
 | |
|             selectedIds.push({
 | |
|                 id: id
 | |
|             });
 | |
|             selectedItems.push(block);
 | |
|             selectedItemsElements.push($parent);
 | |
|         });
 | |
| 
 | |
|         if (selectedItems.length !== 0) {
 | |
|             var formattedItems = "";
 | |
|             var i = 0;
 | |
|             $.each(selectedItems, function (key, value) {
 | |
|                 formattedItems += ("<code>" + value + "</code>, ");
 | |
|                 i++;
 | |
|                 return i < 5;
 | |
|             });
 | |
| 
 | |
|             formattedItems = formattedItems.slice(0, -2);
 | |
|             if (selectedItems.length > 5) {
 | |
|                 formattedItems += ', and ' + (selectedItems.length - 5) + ' other(s)';
 | |
|             }
 | |
| 
 | |
|             swal({
 | |
|                 type: 'warning',
 | |
|                 title: '',
 | |
|                 text: 'Are you sure you want to delete the following allocations: ' + formattedItems + '?',
 | |
|                 html: true,
 | |
|                 showCancelButton: true,
 | |
|                 showConfirmButton: true,
 | |
|                 closeOnConfirm: false,
 | |
|                 showLoaderOnConfirm: true
 | |
|             }, function () {
 | |
|                 $.ajax({
 | |
|                     method: 'DELETE',
 | |
|                     url: '/admin/nodes/view/' + {{ $node->id }} + '/allocations',
 | |
|                     headers: {'X-CSRF-TOKEN': $('meta[name="_token"]').attr('content')},
 | |
|                     data: JSON.stringify({
 | |
|                         allocations: selectedIds
 | |
|                     }),
 | |
|                     contentType: 'application/json',
 | |
|                     processData: false
 | |
|                 }).done(function () {
 | |
|                     $('#file_listing input:checked').each(function () {
 | |
|                         $(this).prop('checked', false);
 | |
|                     });
 | |
| 
 | |
|                     $.each(selectedItemsElements, function () {
 | |
|                         $(this).addClass('warning').delay(200).fadeOut();
 | |
|                     });
 | |
| 
 | |
|                     swal({
 | |
|                         type: 'success',
 | |
|                         title: 'Allocations Deleted'
 | |
|                     });
 | |
|                 }).fail(function (jqXHR) {
 | |
|                     console.error(jqXHR);
 | |
|                     swal({
 | |
|                         type: 'error',
 | |
|                         title: 'Whoops!',
 | |
|                         html: true,
 | |
|                         text: 'An error occurred while attempting to delete these allocations. Please try again.',
 | |
|                     });
 | |
|                 });
 | |
|             });
 | |
|         } else {
 | |
|             swal({
 | |
|                 type: 'warning',
 | |
|                 title: '',
 | |
|                 text: 'Please select allocation(s) to delete.',
 | |
|             });
 | |
|         }
 | |
|     }
 | |
|     </script>
 | |
| @endsection
 | 
