mirror of https://github.com/portainer/portainer
refactor(global): revert merge with internal (#133)
parent
b5dcdc8807
commit
eefa7ca138
11
README.md
11
README.md
|
@ -101,16 +101,6 @@ You can hide it in the view by starting the ui with:
|
||||||
$ docker run -d -p 9000:9000 --privileged -v /var/run/docker.sock:/var/run/docker.sock cloudinovasi/cloudinovasi-ui -l owner=acme
|
$ docker run -d -p 9000:9000 --privileged -v /var/run/docker.sock:/var/run/docker.sock cloudinovasi/cloudinovasi-ui -l owner=acme
|
||||||
```
|
```
|
||||||
|
|
||||||
### Custom Docker registries support
|
|
||||||
|
|
||||||
You can specify the support of others registries than DockerHub by using the `--registries` or `-r` options and specifying a registry using the format *REGISTRY_NAME=REGISTRY_ADDRESS*.
|
|
||||||
|
|
||||||
For example, if I want the registry 'myCustomRegistry' pointing to *myregistry.domain.com:5000* available in the UI:
|
|
||||||
|
|
||||||
```
|
|
||||||
$ docker run -d -p 9000:9000 --privileged -v /var/run/docker.sock:/var/run/docker.sock cloudinovasi/cloudinovasi-ui -r myCustomRegistry=myregistry.domain.com:5000
|
|
||||||
```
|
|
||||||
|
|
||||||
### Available options
|
### Available options
|
||||||
|
|
||||||
The following options are available for the `ui-for-docker` binary:
|
The following options are available for the `ui-for-docker` binary:
|
||||||
|
@ -120,7 +110,6 @@ The following options are available for the `ui-for-docker` binary:
|
||||||
* `--data`, `-d`: Path to the data folder (default: `"."`)
|
* `--data`, `-d`: Path to the data folder (default: `"."`)
|
||||||
* `--assets`, `-a`: Path to the assets (default: `"."`)
|
* `--assets`, `-a`: Path to the assets (default: `"."`)
|
||||||
* `--swarm`, `-s`: Swarm cluster support (default: `false`)
|
* `--swarm`, `-s`: Swarm cluster support (default: `false`)
|
||||||
* `--registries`, `-r`: Available registries in the UI (format *REGISTRY_NAME=REGISTRY_ADDRESS*)
|
|
||||||
* `--tlsverify`: TLS support (default: `false`)
|
* `--tlsverify`: TLS support (default: `false`)
|
||||||
* `--tlscacert`: Path to the CA (default `/certs/ca.pem`)
|
* `--tlscacert`: Path to the CA (default `/certs/ca.pem`)
|
||||||
* `--tlscert`: Path to the TLS certificate file (default `/certs/cert.pem`)
|
* `--tlscert`: Path to the TLS certificate file (default `/certs/cert.pem`)
|
||||||
|
|
25
api/main.go
25
api/main.go
|
@ -8,17 +8,16 @@ import (
|
||||||
func main() {
|
func main() {
|
||||||
kingpin.Version("1.6.0")
|
kingpin.Version("1.6.0")
|
||||||
var (
|
var (
|
||||||
endpoint = kingpin.Flag("host", "Dockerd endpoint").Default("unix:///var/run/docker.sock").Short('H').String()
|
endpoint = kingpin.Flag("host", "Dockerd endpoint").Default("unix:///var/run/docker.sock").Short('H').String()
|
||||||
addr = kingpin.Flag("bind", "Address and port to serve UI For Docker").Default(":9000").Short('p').String()
|
addr = kingpin.Flag("bind", "Address and port to serve UI For Docker").Default(":9000").Short('p').String()
|
||||||
assets = kingpin.Flag("assets", "Path to the assets").Default(".").Short('a').String()
|
assets = kingpin.Flag("assets", "Path to the assets").Default(".").Short('a').String()
|
||||||
data = kingpin.Flag("data", "Path to the data").Default(".").Short('d').String()
|
data = kingpin.Flag("data", "Path to the data").Default(".").Short('d').String()
|
||||||
tlsverify = kingpin.Flag("tlsverify", "TLS support").Default("false").Bool()
|
tlsverify = kingpin.Flag("tlsverify", "TLS support").Default("false").Bool()
|
||||||
tlscacert = kingpin.Flag("tlscacert", "Path to the CA").Default("/certs/ca.pem").String()
|
tlscacert = kingpin.Flag("tlscacert", "Path to the CA").Default("/certs/ca.pem").String()
|
||||||
tlscert = kingpin.Flag("tlscert", "Path to the TLS certificate file").Default("/certs/cert.pem").String()
|
tlscert = kingpin.Flag("tlscert", "Path to the TLS certificate file").Default("/certs/cert.pem").String()
|
||||||
tlskey = kingpin.Flag("tlskey", "Path to the TLS key").Default("/certs/key.pem").String()
|
tlskey = kingpin.Flag("tlskey", "Path to the TLS key").Default("/certs/key.pem").String()
|
||||||
swarm = kingpin.Flag("swarm", "Swarm cluster support").Default("false").Short('s').Bool()
|
swarm = kingpin.Flag("swarm", "Swarm cluster support").Default("false").Short('s').Bool()
|
||||||
labels = pairs(kingpin.Flag("hide-label", "Hide containers with a specific label in the UI").Short('l'))
|
labels = pairs(kingpin.Flag("hide-label", "Hide containers with a specific label in the UI").Short('l'))
|
||||||
registries = pairs(kingpin.Flag("registries", "Supported Docker registries").Short('r'))
|
|
||||||
logo = kingpin.Flag("logo", "URL for the logo displayed in the UI").String()
|
logo = kingpin.Flag("logo", "URL for the logo displayed in the UI").String()
|
||||||
)
|
)
|
||||||
kingpin.Parse()
|
kingpin.Parse()
|
||||||
|
@ -38,10 +37,6 @@ func main() {
|
||||||
settings := &Settings{
|
settings := &Settings{
|
||||||
Swarm: *swarm,
|
Swarm: *swarm,
|
||||||
HiddenLabels: *labels,
|
HiddenLabels: *labels,
|
||||||
<<<<<<< HEAD
|
|
||||||
Registries: *registries,
|
|
||||||
=======
|
|
||||||
>>>>>>> feat107-push-registry
|
|
||||||
Logo: *logo,
|
Logo: *logo,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,6 @@ import (
|
||||||
type Settings struct {
|
type Settings struct {
|
||||||
Swarm bool `json:"swarm"`
|
Swarm bool `json:"swarm"`
|
||||||
HiddenLabels pairList `json:"hiddenLabels"`
|
HiddenLabels pairList `json:"hiddenLabels"`
|
||||||
Registries pairList `json:"registries"`
|
|
||||||
Logo string `json:"logo"`
|
Logo string `json:"logo"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,7 +21,9 @@ angular.module('uifordocker', [
|
||||||
'swarm',
|
'swarm',
|
||||||
'network',
|
'network',
|
||||||
'networks',
|
'networks',
|
||||||
'volumes'])
|
'createNetwork',
|
||||||
|
'volumes',
|
||||||
|
'createVolume'])
|
||||||
.config(['$stateProvider', '$urlRouterProvider', '$httpProvider', function ($stateProvider, $urlRouterProvider, $httpProvider) {
|
.config(['$stateProvider', '$urlRouterProvider', '$httpProvider', function ($stateProvider, $urlRouterProvider, $httpProvider) {
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,74 @@
|
||||||
|
angular.module('createNetwork', [])
|
||||||
|
.controller('CreateNetworkController', ['$scope', '$state', 'Messages', 'Network', 'errorMsgFilter',
|
||||||
|
function ($scope, $state, Messages, Network, errorMsgFilter) {
|
||||||
|
$scope.formValues = {
|
||||||
|
DriverOptions: [],
|
||||||
|
Subnet: '',
|
||||||
|
Gateway: ''
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.config = {
|
||||||
|
Driver: 'bridge',
|
||||||
|
CheckDuplicate: true,
|
||||||
|
Internal: false,
|
||||||
|
IPAM: {
|
||||||
|
Config: []
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.addDriverOption = function() {
|
||||||
|
$scope.formValues.DriverOptions.push({ name: '', value: '' });
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.removeDriverOption = function(index) {
|
||||||
|
$scope.formValues.DriverOptions.splice(index, 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
function createNetwork(config) {
|
||||||
|
$('#createNetworkSpinner').show();
|
||||||
|
Network.create(config, function (d) {
|
||||||
|
if (d.Id) {
|
||||||
|
Messages.send("Network created", d.Id);
|
||||||
|
$('#createNetworkSpinner').hide();
|
||||||
|
$state.go('networks', {}, {reload: true});
|
||||||
|
} else {
|
||||||
|
$('#createNetworkSpinner').hide();
|
||||||
|
Messages.error('Unable to create network', errorMsgFilter(d));
|
||||||
|
}
|
||||||
|
}, function (e) {
|
||||||
|
$('#createNetworkSpinner').hide();
|
||||||
|
Messages.error('Unable to create network', e.data);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareIPAMConfiguration(config) {
|
||||||
|
if ($scope.formValues.Subnet) {
|
||||||
|
var ipamConfig = {};
|
||||||
|
ipamConfig.Subnet = $scope.formValues.Subnet;
|
||||||
|
if ($scope.formValues.Gateway) {
|
||||||
|
ipamConfig.Gateway = $scope.formValues.Gateway ;
|
||||||
|
}
|
||||||
|
config.IPAM.Config.push(ipamConfig);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareDriverOptions(config) {
|
||||||
|
var options = {};
|
||||||
|
$scope.formValues.DriverOptions.forEach(function (option) {
|
||||||
|
options[option.name] = option.value;
|
||||||
|
});
|
||||||
|
config.Options = options;
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareConfiguration() {
|
||||||
|
var config = angular.copy($scope.config);
|
||||||
|
prepareIPAMConfiguration(config);
|
||||||
|
prepareDriverOptions(config);
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.create = function () {
|
||||||
|
var config = prepareConfiguration();
|
||||||
|
createNetwork(config);
|
||||||
|
};
|
||||||
|
}]);
|
|
@ -0,0 +1,95 @@
|
||||||
|
<rd-header>
|
||||||
|
<rd-header-title title="Create network"></rd-header-title>
|
||||||
|
<rd-header-content>
|
||||||
|
Networks > Add network
|
||||||
|
</rd-header-content>
|
||||||
|
</rd-header>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12 col-md-12 col-xs-12">
|
||||||
|
<rd-widget>
|
||||||
|
<rd-widget-body>
|
||||||
|
<form class="form-horizontal">
|
||||||
|
<!-- name-input -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="network_name" class="col-sm-1 control-label text-left">Name</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<input type="text" class="form-control" ng-model="config.Name" id="network_name" placeholder="e.g. myNetwork">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !name-input -->
|
||||||
|
<!-- subnet-gateway-inputs -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="network_subnet" class="col-sm-1 control-label text-left">Subnet</label>
|
||||||
|
<div class="col-sm-5">
|
||||||
|
<input type="text" class="form-control" ng-model="formValues.Subnet" id="network_subnet" placeholder="e.g. 172.20.0.0/16">
|
||||||
|
</div>
|
||||||
|
<label for="network_gateway" class="col-sm-1 control-label text-left">Gateway</label>
|
||||||
|
<div class="col-sm-5">
|
||||||
|
<input type="text" class="form-control" ng-model="formValues.Gateway" id="network_gateway" placeholder="e.g. 172.20.10.11">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !subnet-gateway-inputs -->
|
||||||
|
<!-- driver-input -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="network_driver" class="col-sm-1 control-label text-left">Driver</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<input type="text" class="form-control" ng-model="config.Driver" id="network_driver" placeholder="e.g. driverName">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !driver-input -->
|
||||||
|
<!-- driver-options -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="network_driveropts" class="col-sm-1 control-label text-left">Driver options</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<span class="label label-default clickable" ng-click="addDriverOption()">
|
||||||
|
<i class="fa fa-plus-circle" aria-hidden="true"></i> driver option
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<!-- driver-options-input-list -->
|
||||||
|
<div class="col-sm-offset-1 col-sm-11 form-inline" style="margin-top: 10px;">
|
||||||
|
<div ng-repeat="option in formValues.DriverOptions" style="margin-top: 2px;">
|
||||||
|
<div class="input-group col-sm-5 input-group-sm">
|
||||||
|
<span class="input-group-addon">name</span>
|
||||||
|
<input type="text" class="form-control" ng-model="option.name" placeholder="e.g. com.docker.network.bridge.enable_icc">
|
||||||
|
</div>
|
||||||
|
<div class="input-group col-sm-5 input-group-sm">
|
||||||
|
<span class="input-group-addon">value</span>
|
||||||
|
<input type="text" class="form-control" ng-model="option.value" placeholder="e.g. true">
|
||||||
|
<span class="input-group-btn">
|
||||||
|
<button class="btn btn-default" type="button" ng-click="removeDriverOption($index)">
|
||||||
|
<i class="fa fa-minus" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !driver-options-input-list -->
|
||||||
|
</div>
|
||||||
|
<!-- !driver-options -->
|
||||||
|
<!-- internal -->
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="col-sm-12">
|
||||||
|
<div class="checkbox">
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" ng-model="config.Internal"> Restrict external access to the network
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !internal -->
|
||||||
|
</form>
|
||||||
|
</rd-widget-body>
|
||||||
|
</rd-widget>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12 col-md-12 col-xs-12" style="text-align: center;">
|
||||||
|
<div>
|
||||||
|
<i id="createNetworkSpinner" class="fa fa-cog fa-3x fa-spin" style="margin-bottom: 5px; display: none;"></i>
|
||||||
|
</div>
|
||||||
|
<button type="button" class="btn btn-default btn-lg" ng-click="create()">Create</button>
|
||||||
|
<a type="button" class="btn btn-default btn-lg" ui-sref="networks">Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
|
@ -0,0 +1,56 @@
|
||||||
|
angular.module('createVolume', [])
|
||||||
|
.controller('CreateVolumeController', ['$scope', '$state', 'Volume', 'Messages', 'errorMsgFilter',
|
||||||
|
function ($scope, $state, Volume, Messages, errorMsgFilter) {
|
||||||
|
|
||||||
|
$scope.formValues = {
|
||||||
|
DriverOptions: []
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.config = {
|
||||||
|
Driver: 'local'
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.addDriverOption = function() {
|
||||||
|
$scope.formValues.DriverOptions.push({ name: '', value: '' });
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.removeDriverOption = function(index) {
|
||||||
|
$scope.formValues.DriverOptions.splice(index, 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
function createVolume(config) {
|
||||||
|
$('#createVolumeSpinner').show();
|
||||||
|
Volume.create(config, function (d) {
|
||||||
|
if (d.Name) {
|
||||||
|
Messages.send("Volume created", d.Name);
|
||||||
|
$('#createVolumeSpinner').hide();
|
||||||
|
$state.go('volumes', {}, {reload: true});
|
||||||
|
} else {
|
||||||
|
$('#createVolumeSpinner').hide();
|
||||||
|
Messages.error('Unable to create volume', errorMsgFilter(d));
|
||||||
|
}
|
||||||
|
}, function (e) {
|
||||||
|
$('#createVolumeSpinner').hide();
|
||||||
|
Messages.error('Unable to create volume', e.data);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareDriverOptions(config) {
|
||||||
|
var options = {};
|
||||||
|
$scope.formValues.DriverOptions.forEach(function (option) {
|
||||||
|
options[option.name] = option.value;
|
||||||
|
});
|
||||||
|
config.DriverOpts = options;
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareConfiguration() {
|
||||||
|
var config = angular.copy($scope.config);
|
||||||
|
prepareDriverOptions(config);
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.create = function () {
|
||||||
|
var config = prepareConfiguration();
|
||||||
|
createVolume(config);
|
||||||
|
};
|
||||||
|
}]);
|
|
@ -0,0 +1,72 @@
|
||||||
|
<rd-header>
|
||||||
|
<rd-header-title title="Create volume"></rd-header-title>
|
||||||
|
<rd-header-content>
|
||||||
|
Volumes > Add volume
|
||||||
|
</rd-header-content>
|
||||||
|
</rd-header>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12 col-md-12 col-xs-12">
|
||||||
|
<rd-widget>
|
||||||
|
<rd-widget-body>
|
||||||
|
<form class="form-horizontal">
|
||||||
|
<!-- name-input -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="volume_name" class="col-sm-1 control-label text-left">Name</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<input type="text" class="form-control" ng-model="config.Name" id="volume_name" placeholder="e.g. myVolume">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !name-input -->
|
||||||
|
<!-- driver-input -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="volume_driver" class="col-sm-1 control-label text-left">Driver</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<input type="text" class="form-control" ng-model="config.Driver" id="volume_driver" placeholder="e.g. driverName">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !driver-input -->
|
||||||
|
<!-- driver-options -->
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="volume_driveropts" class="col-sm-1 control-label text-left">Driver options</label>
|
||||||
|
<div class="col-sm-11">
|
||||||
|
<span class="label label-default clickable" ng-click="addDriverOption()">
|
||||||
|
<i class="fa fa-plus-circle" aria-hidden="true"></i> driver option
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<!-- driver-options-input-list -->
|
||||||
|
<div class="col-sm-offset-1 col-sm-11 form-inline" style="margin-top: 10px;">
|
||||||
|
<div ng-repeat="option in formValues.DriverOptions" style="margin-top: 2px;">
|
||||||
|
<div class="input-group col-sm-5 input-group-sm">
|
||||||
|
<span class="input-group-addon">name</span>
|
||||||
|
<input type="text" class="form-control" ng-model="option.name" placeholder="e.g. mountpoint">
|
||||||
|
</div>
|
||||||
|
<div class="input-group col-sm-5 input-group-sm">
|
||||||
|
<span class="input-group-addon">value</span>
|
||||||
|
<input type="text" class="form-control" ng-model="option.value" placeholder="e.g. /path/on/host">
|
||||||
|
<span class="input-group-btn">
|
||||||
|
<button class="btn btn-default" type="button" ng-click="removeDriverOption($index)">
|
||||||
|
<i class="fa fa-minus" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- !driver-options-input-list -->
|
||||||
|
</div>
|
||||||
|
<!-- !driver-options -->
|
||||||
|
</form>
|
||||||
|
</rd-widget-body>
|
||||||
|
</rd-widget>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-lg-12 col-md-12 col-xs-12" style="text-align: center;">
|
||||||
|
<div>
|
||||||
|
<i id="createVolumeSpinner" class="fa fa-cog fa-3x fa-spin" style="margin-bottom: 5px; display: none;"></i>
|
||||||
|
</div>
|
||||||
|
<button type="button" class="btn btn-default btn-lg" ng-click="create()">Create</button>
|
||||||
|
<a type="button" class="btn btn-default btn-lg" ui-sref="volumes">Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
|
@ -22,10 +22,7 @@
|
||||||
</div>
|
</div>
|
||||||
<label for="image_registry" class="col-sm-1 control-label text-left">Registry</label>
|
<label for="image_registry" class="col-sm-1 control-label text-left">Registry</label>
|
||||||
<div class="col-sm-3">
|
<div class="col-sm-3">
|
||||||
<select class="selectpicker form-control" ng-model="config.Registry">
|
<input type="text" class="form-control" ng-model="config.Registry" id="image_registry" placeholder="leave empty to use DockerHub">
|
||||||
<option value="">Docker Hub</option>
|
|
||||||
<option ng-repeat="registry in availableRegistries" ng-value="registry.value">{{ registry.name }}</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<!-- !name-and-registry-inputs -->
|
<!-- !name-and-registry-inputs -->
|
||||||
|
@ -91,6 +88,13 @@
|
||||||
<span ng-show="sortType == 'VirtualSize' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'VirtualSize' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
</th>
|
</th>
|
||||||
|
<th>
|
||||||
|
<a ui-sref="images" ng-click="order('Created')">
|
||||||
|
Created
|
||||||
|
<span ng-show="sortType == 'Created' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
|
<span ng-show="sortType == 'Created' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
|
@ -101,11 +105,12 @@
|
||||||
<span class="label label-primary image-tag" ng-repeat="tag in (image|repotags)">{{ tag }}</span>
|
<span class="label label-primary image-tag" ng-repeat="tag in (image|repotags)">{{ tag }}</span>
|
||||||
</td>
|
</td>
|
||||||
<td>{{ image.VirtualSize|humansize }}</td>
|
<td>{{ image.VirtualSize|humansize }}</td>
|
||||||
|
<td>{{ image.Created|getisodatefromtimestamp }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</rd-widget-body>
|
</rd-widget-body>
|
||||||
<rd-widget>
|
<rd-widget>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -40,7 +40,7 @@ function ($scope, $state, Config, Image, Messages) {
|
||||||
$scope.pullImage = function() {
|
$scope.pullImage = function() {
|
||||||
$('#pullImageSpinner').show();
|
$('#pullImageSpinner').show();
|
||||||
var image = _.toLower($scope.config.Image);
|
var image = _.toLower($scope.config.Image);
|
||||||
var registry = $scope.config.Registry;
|
var registry = _.toLower($scope.config.Registry);
|
||||||
var imageConfig = createImageConfig(image, registry);
|
var imageConfig = createImageConfig(image, registry);
|
||||||
Image.create(imageConfig, function (data) {
|
Image.create(imageConfig, function (data) {
|
||||||
var err = data.length > 0 && data[data.length - 1].hasOwnProperty('error');
|
var err = data.length > 0 && data[data.length - 1].hasOwnProperty('error');
|
||||||
|
|
|
@ -7,64 +7,6 @@
|
||||||
<rd-header-content>Networks</rd-header-content>
|
<rd-header-content>Networks</rd-header-content>
|
||||||
</rd-header>
|
</rd-header>
|
||||||
|
|
||||||
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-lg-12 col-md-12 col-xs-12">
|
|
||||||
<rd-widget>
|
|
||||||
<rd-widget-header icon="fa-plus" title="Add a shared network">
|
|
||||||
</rd-widget-header>
|
|
||||||
<rd-widget-body>
|
|
||||||
<form class="form-horizontal">
|
|
||||||
<!-- name-input -->
|
|
||||||
<div class="form-group">
|
|
||||||
<label for="network_name" class="col-sm-1 control-label text-left">Name</label>
|
|
||||||
<div class="col-sm-11">
|
|
||||||
<input type="text" class="form-control" ng-model="config.Name" id="network_name" placeholder="e.g. myNetwork">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !name-input -->
|
|
||||||
<!-- advanced-settings-input -->
|
|
||||||
<div class="form-group">
|
|
||||||
<div class="col-sm-12">
|
|
||||||
<div class="checkbox">
|
|
||||||
<label>
|
|
||||||
<input type="checkbox" ng-model="state.advancedSettings"> Show advanced settings
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !advanced-settings-input -->
|
|
||||||
<!-- subnet-gateway-inputs -->
|
|
||||||
<div class="form-group" ng-if="state.advancedSettings">
|
|
||||||
<label for="network_subnet" class="col-sm-1 control-label text-left">Subnet</label>
|
|
||||||
<div class="col-sm-5">
|
|
||||||
<input type="text" class="form-control" ng-model="formValues.Subnet" id="network_subnet" placeholder="e.g. 172.20.0.0/16">
|
|
||||||
</div>
|
|
||||||
<label for="network_gateway" class="col-sm-1 control-label text-left">Gateway</label>
|
|
||||||
<div class="col-sm-5">
|
|
||||||
<input type="text" class="form-control" ng-model="formValues.Gateway" id="network_gateway" placeholder="e.g. 172.20.10.11">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !subnet-gateway-inputs -->
|
|
||||||
<!-- tag-note -->
|
|
||||||
<div class="form-group" ng-if="swarm">
|
|
||||||
<div class="col-sm-12">
|
|
||||||
<span class="small text-muted">Note: The network will be created using the overlay driver and will allow containers to communicate across the hosts of your cluster.</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !tag-note -->
|
|
||||||
<div class="form-group">
|
|
||||||
<div class="col-sm-12">
|
|
||||||
<button type="button" class="btn btn-default btn-sm" ng-disabled="!config.Name" ng-click="createNetwork()">Create</button>
|
|
||||||
<i id="createNetworkSpinner" class="fa fa-cog fa-spin" style="margin-left: 5px; display: none;"></i>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</rd-widget-body>
|
|
||||||
</rd-widget>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<rd-widget>
|
<rd-widget>
|
||||||
<rd-widget-header icon="fa-sitemap" title="Networks">
|
<rd-widget-header icon="fa-sitemap" title="Networks">
|
||||||
|
@ -75,6 +17,7 @@
|
||||||
<rd-widget-taskbar classes="col-lg-12">
|
<rd-widget-taskbar classes="col-lg-12">
|
||||||
<div class="pull-left">
|
<div class="pull-left">
|
||||||
<button type="button" class="btn btn-danger" ng-click="removeAction()" ng-disabled="!state.selectedItemCount">Remove</button>
|
<button type="button" class="btn btn-danger" ng-click="removeAction()" ng-disabled="!state.selectedItemCount">Remove</button>
|
||||||
|
<a class="btn btn-default" type="button" ui-sref="actions.create.network">Add network</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="pull-right">
|
<div class="pull-right">
|
||||||
<input type="text" id="filter" ng-model="state.filter" placeholder="Filter..." class="form-control input-sm" />
|
<input type="text" id="filter" ng-model="state.filter" placeholder="Filter..." class="form-control input-sm" />
|
||||||
|
@ -93,6 +36,13 @@
|
||||||
<span ng-show="sortType == 'Name' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'Name' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
</th>
|
</th>
|
||||||
|
<th>
|
||||||
|
<a ui-sref="networks" ng-click="order('Id')">
|
||||||
|
Id
|
||||||
|
<span ng-show="sortType == 'Id' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
|
<span ng-show="sortType == 'Id' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
<th>
|
<th>
|
||||||
<a ui-sref="networks" ng-click="order('Scope')">
|
<a ui-sref="networks" ng-click="order('Scope')">
|
||||||
Scope
|
Scope
|
||||||
|
@ -100,16 +50,30 @@
|
||||||
<span ng-show="sortType == 'Scope' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'Scope' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
</th>
|
</th>
|
||||||
|
<th>
|
||||||
|
<a ui-sref="networks" ng-click="order('Driver')">
|
||||||
|
Driver
|
||||||
|
<span ng-show="sortType == 'Driver' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
|
<span ng-show="sortType == 'Driver' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
<a ui-sref="networks" ng-click="order('IPAM.Driver')">
|
||||||
|
IPAM Driver
|
||||||
|
<span ng-show="sortType == 'IPAM.Driver' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
|
<span ng-show="sortType == 'IPAM.Driver' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
<th>
|
<th>
|
||||||
<a ui-sref="networks" ng-click="order('IPAM.Config[0].Subnet')">
|
<a ui-sref="networks" ng-click="order('IPAM.Config[0].Subnet')">
|
||||||
Subnet
|
IPAM Subnet
|
||||||
<span ng-show="sortType == 'IPAM.Config[0].Subnet' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
<span ng-show="sortType == 'IPAM.Config[0].Subnet' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
<span ng-show="sortType == 'IPAM.Config[0].Subnet' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'IPAM.Config[0].Subnet' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
<a ui-sref="networks" ng-click="order('IPAM.Config[0].Gateway')">
|
<a ui-sref="networks" ng-click="order('IPAM.Config[0].Gateway')">
|
||||||
Gateway
|
IPAM Gateway
|
||||||
<span ng-show="sortType == 'IPAM.Config[0].Gateway' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
<span ng-show="sortType == 'IPAM.Config[0].Gateway' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
<span ng-show="sortType == 'IPAM.Config[0].Gateway' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'IPAM.Config[0].Gateway' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
|
@ -120,7 +84,10 @@
|
||||||
<tr ng-repeat="network in ( state.filteredNetworks = (networks | filter:state.filter | orderBy:sortType:sortReverse))">
|
<tr ng-repeat="network in ( state.filteredNetworks = (networks | filter:state.filter | orderBy:sortType:sortReverse))">
|
||||||
<td><input type="checkbox" ng-model="network.Checked" ng-change="selectItem(network)"/></td>
|
<td><input type="checkbox" ng-model="network.Checked" ng-change="selectItem(network)"/></td>
|
||||||
<td><a ui-sref="network({id: network.Id})">{{ network.Name|truncate:40}}</a></td>
|
<td><a ui-sref="network({id: network.Id})">{{ network.Name|truncate:40}}</a></td>
|
||||||
|
<td>{{ network.Id }}</td>
|
||||||
<td>{{ network.Scope }}</td>
|
<td>{{ network.Scope }}</td>
|
||||||
|
<td>{{ network.Driver }}</td>
|
||||||
|
<td>{{ network.IPAM.Driver }}</td>
|
||||||
<td>{{ network.IPAM.Config[0].Subnet ? network.IPAM.Config[0].Subnet : '-' }}</td>
|
<td>{{ network.IPAM.Config[0].Subnet ? network.IPAM.Config[0].Subnet : '-' }}</td>
|
||||||
<td>{{ network.IPAM.Config[0].Gateway ? network.IPAM.Config[0].Gateway : '-' }}</td>
|
<td>{{ network.IPAM.Config[0].Gateway ? network.IPAM.Config[0].Gateway : '-' }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
|
@ -4,7 +4,7 @@ function ($scope, $state, Network, Config, Messages, errorMsgFilter) {
|
||||||
$scope.state = {};
|
$scope.state = {};
|
||||||
$scope.state.selectedItemCount = 0;
|
$scope.state.selectedItemCount = 0;
|
||||||
$scope.state.advancedSettings = false;
|
$scope.state.advancedSettings = false;
|
||||||
$scope.sortType = 'Scope';
|
$scope.sortType = 'Name';
|
||||||
$scope.sortReverse = false;
|
$scope.sortReverse = false;
|
||||||
|
|
||||||
$scope.formValues = {
|
$scope.formValues = {
|
||||||
|
@ -32,44 +32,6 @@ function ($scope, $state, Network, Config, Messages, errorMsgFilter) {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
function prepareIPAMConfiguration(config) {
|
|
||||||
if ($scope.formValues.Subnet) {
|
|
||||||
var ipamConfig = {};
|
|
||||||
ipamConfig.Subnet = $scope.formValues.Subnet;
|
|
||||||
if ($scope.formValues.Gateway) {
|
|
||||||
ipamConfig.Gateway = $scope.formValues.Gateway ;
|
|
||||||
}
|
|
||||||
config.IPAM.Config.push(ipamConfig);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function prepareNetworkConfiguration() {
|
|
||||||
var config = angular.copy($scope.config);
|
|
||||||
prepareIPAMConfiguration(config);
|
|
||||||
if ($scope.swarm) {
|
|
||||||
config.Driver = 'overlay';
|
|
||||||
}
|
|
||||||
return config;
|
|
||||||
}
|
|
||||||
|
|
||||||
$scope.createNetwork = function() {
|
|
||||||
$('#createNetworkSpinner').show();
|
|
||||||
var config = prepareNetworkConfiguration();
|
|
||||||
Network.create(config, function (d) {
|
|
||||||
if (d.Id) {
|
|
||||||
Messages.send("Network created", d.Id);
|
|
||||||
$('#createNetworkSpinner').hide();
|
|
||||||
$state.go('networks', {}, {reload: true});
|
|
||||||
} else {
|
|
||||||
$('#createNetworkSpinner').hide();
|
|
||||||
Messages.error('Unable to create network', errorMsgFilter(d));
|
|
||||||
}
|
|
||||||
}, function (e) {
|
|
||||||
$('#createNetworkSpinner').hide();
|
|
||||||
Messages.error('Unable to create network', e.data);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
$scope.removeAction = function () {
|
$scope.removeAction = function () {
|
||||||
$('#loadNetworksSpinner').show();
|
$('#loadNetworksSpinner').show();
|
||||||
var counter = 0;
|
var counter = 0;
|
||||||
|
@ -111,8 +73,5 @@ function ($scope, $state, Network, Config, Messages, errorMsgFilter) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Config.$promise.then(function (c) {
|
fetchNetworks();
|
||||||
$scope.swarm = c.swarm;
|
|
||||||
fetchNetworks();
|
|
||||||
});
|
|
||||||
}]);
|
}]);
|
||||||
|
|
|
@ -7,40 +7,6 @@
|
||||||
<rd-header-content>Volumes</rd-header-content>
|
<rd-header-content>Volumes</rd-header-content>
|
||||||
</rd-header>
|
</rd-header>
|
||||||
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-lg-12 col-md-12 col-xs-12">
|
|
||||||
<rd-widget>
|
|
||||||
<rd-widget-header icon="fa-plus" title="Add a persistent volume">
|
|
||||||
</rd-widget-header>
|
|
||||||
<rd-widget-body>
|
|
||||||
<form class="form-horizontal">
|
|
||||||
<!-- name-input -->
|
|
||||||
<div class="form-group">
|
|
||||||
<label for="volume_name" class="col-sm-1 control-label text-left">Name</label>
|
|
||||||
<div class="col-sm-11">
|
|
||||||
<input type="text" class="form-control" ng-model="config.Name" id="volume_name" placeholder="e.g. mysql-data">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !name-input -->
|
|
||||||
<!-- tag-note -->
|
|
||||||
<div class="form-group">
|
|
||||||
<div class="col-sm-12">
|
|
||||||
<span class="small text-muted">Note: The volume will be created in our persisted storage and will be available across all the hosts of your cluster.</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- !tag-note -->
|
|
||||||
<div class="form-group">
|
|
||||||
<div class="col-sm-12">
|
|
||||||
<button type="button" class="btn btn-default btn-sm" ng-disabled="!config.Name" ng-click="createVolume()">Create</button>
|
|
||||||
<i id="createVolumeSpinner" class="fa fa-cog fa-spin" style="margin-left: 5px; display: none;"></i>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</rd-widget-body>
|
|
||||||
</rd-widget>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<rd-widget>
|
<rd-widget>
|
||||||
<rd-widget-header icon="fa-cubes" title="Volumes">
|
<rd-widget-header icon="fa-cubes" title="Volumes">
|
||||||
|
@ -51,6 +17,7 @@
|
||||||
<rd-widget-taskbar classes="col-lg-12">
|
<rd-widget-taskbar classes="col-lg-12">
|
||||||
<div class="pull-left">
|
<div class="pull-left">
|
||||||
<button type="button" class="btn btn-danger" ng-click="removeAction()" ng-disabled="!state.selectedItemCount">Remove</button>
|
<button type="button" class="btn btn-danger" ng-click="removeAction()" ng-disabled="!state.selectedItemCount">Remove</button>
|
||||||
|
<a class="btn btn-default" type="button" ui-sref="actions.create.volume">Add volume</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="pull-right">
|
<div class="pull-right">
|
||||||
<input type="text" id="filter" ng-model="state.filter" placeholder="Filter..." class="form-control input-sm" />
|
<input type="text" id="filter" ng-model="state.filter" placeholder="Filter..." class="form-control input-sm" />
|
||||||
|
@ -76,6 +43,13 @@
|
||||||
<span ng-show="sortType == 'Driver' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
<span ng-show="sortType == 'Driver' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
</a>
|
</a>
|
||||||
</th>
|
</th>
|
||||||
|
<th>
|
||||||
|
<a ui-sref="volumes" ng-click="order('Mountpoint')">
|
||||||
|
Mountpoint
|
||||||
|
<span ng-show="sortType == 'Mountpoint' && !sortReverse" class="glyphicon glyphicon-chevron-down"></span>
|
||||||
|
<span ng-show="sortType == 'Mountpoint' && sortReverse" class="glyphicon glyphicon-chevron-up"></span>
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
|
@ -83,10 +57,11 @@
|
||||||
<td><input type="checkbox" ng-model="volume.Checked" ng-change="selectItem(volume)"/></td>
|
<td><input type="checkbox" ng-model="volume.Checked" ng-change="selectItem(volume)"/></td>
|
||||||
<td>{{ volume.Name|truncate:50 }}</td>
|
<td>{{ volume.Name|truncate:50 }}</td>
|
||||||
<td>{{ volume.Driver }}</td>
|
<td>{{ volume.Driver }}</td>
|
||||||
|
<td>{{ volume.Mountpoint }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</rd-widget-body>
|
</rd-widget-body>
|
||||||
<rd-widget>
|
<rd-widget>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -3,7 +3,7 @@ angular.module('volumes', [])
|
||||||
function ($scope, $state, Volume, Messages, errorMsgFilter) {
|
function ($scope, $state, Volume, Messages, errorMsgFilter) {
|
||||||
$scope.state = {};
|
$scope.state = {};
|
||||||
$scope.state.selectedItemCount = 0;
|
$scope.state.selectedItemCount = 0;
|
||||||
$scope.sortType = 'Driver';
|
$scope.sortType = 'Name';
|
||||||
$scope.sortReverse = true;
|
$scope.sortReverse = true;
|
||||||
|
|
||||||
$scope.config = {
|
$scope.config = {
|
||||||
|
@ -23,32 +23,6 @@ function ($scope, $state, Volume, Messages, errorMsgFilter) {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
function prepareVolumeConfiguration() {
|
|
||||||
var config = angular.copy($scope.config);
|
|
||||||
config.Driver = 'local-persist';
|
|
||||||
config.DriverOpts = {};
|
|
||||||
config.DriverOpts.mountpoint = '/volume/' + config.Name;
|
|
||||||
return config;
|
|
||||||
}
|
|
||||||
|
|
||||||
$scope.createVolume = function() {
|
|
||||||
$('#createVolumeSpinner').show();
|
|
||||||
var config = prepareVolumeConfiguration();
|
|
||||||
Volume.create(config, function (d) {
|
|
||||||
if (d.Name) {
|
|
||||||
Messages.send("Volume created", d.Name);
|
|
||||||
$('#createVolumeSpinner').hide();
|
|
||||||
$state.go('volumes', {}, {reload: true});
|
|
||||||
} else {
|
|
||||||
$('#createVolumeSpinner').hide();
|
|
||||||
Messages.error('Unable to create volume', errorMsgFilter(d));
|
|
||||||
}
|
|
||||||
}, function (e) {
|
|
||||||
$('#createVolumeSpinner').hide();
|
|
||||||
Messages.error('Unable to create volume', e.data);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
$scope.removeAction = function () {
|
$scope.removeAction = function () {
|
||||||
$('#loadVolumesSpinner').show();
|
$('#loadVolumesSpinner').show();
|
||||||
var counter = 0;
|
var counter = 0;
|
||||||
|
@ -77,7 +51,7 @@ function ($scope, $state, Volume, Messages, errorMsgFilter) {
|
||||||
function fetchVolumes() {
|
function fetchVolumes() {
|
||||||
$('#loadVolumesSpinner').show();
|
$('#loadVolumesSpinner').show();
|
||||||
Volume.query({}, function (d) {
|
Volume.query({}, function (d) {
|
||||||
$scope.volumes = _.uniqBy(d.Volumes, 'Name');
|
$scope.volumes = d.Volumes;
|
||||||
$('#loadVolumesSpinner').hide();
|
$('#loadVolumesSpinner').hide();
|
||||||
}, function (e) {
|
}, function (e) {
|
||||||
Messages.error("Failure", e.data);
|
Messages.error("Failure", e.data);
|
||||||
|
|
Loading…
Reference in New Issue