2015-01-22 17:46:38 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2015-01-22 17:46:38 +00:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package kubectl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2015-08-12 13:33:08 +00:00
|
|
|
"reflect"
|
2015-01-22 17:46:38 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/api"
|
2015-10-09 22:04:41 +00:00
|
|
|
"k8s.io/kubernetes/pkg/apis/extensions"
|
2015-08-13 19:01:50 +00:00
|
|
|
client "k8s.io/kubernetes/pkg/client/unversioned"
|
|
|
|
"k8s.io/kubernetes/pkg/client/unversioned/testclient"
|
2015-08-12 13:33:08 +00:00
|
|
|
"k8s.io/kubernetes/pkg/runtime"
|
2015-01-22 17:46:38 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestReplicationControllerStop(t *testing.T) {
|
2015-06-30 01:29:53 +00:00
|
|
|
name := "foo"
|
|
|
|
ns := "default"
|
2015-08-12 13:33:08 +00:00
|
|
|
tests := []struct {
|
|
|
|
Name string
|
|
|
|
Objs []runtime.Object
|
|
|
|
StopError error
|
|
|
|
StopMessage string
|
|
|
|
ExpectedActions []string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Name: "OnlyOneRC",
|
|
|
|
Objs: []runtime.Object{
|
|
|
|
&api.ReplicationController{ // GET
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
&api.ReplicationControllerList{ // LIST
|
|
|
|
Items: []api.ReplicationController{
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
StopError: nil,
|
|
|
|
StopMessage: "foo stopped",
|
|
|
|
ExpectedActions: []string{"get", "list", "get", "update", "get", "get", "delete"},
|
2015-06-30 01:29:53 +00:00
|
|
|
},
|
2015-08-12 13:33:08 +00:00
|
|
|
{
|
|
|
|
Name: "NoOverlapping",
|
|
|
|
Objs: []runtime.Object{
|
|
|
|
&api.ReplicationController{ // GET
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
&api.ReplicationControllerList{ // LIST
|
|
|
|
Items: []api.ReplicationController{
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: "baz",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k3": "v3"}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
StopError: nil,
|
|
|
|
StopMessage: "foo stopped",
|
|
|
|
ExpectedActions: []string{"get", "list", "get", "update", "get", "get", "delete"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "OverlappingError",
|
|
|
|
Objs: []runtime.Object{
|
|
|
|
|
|
|
|
&api.ReplicationController{ // GET
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
&api.ReplicationControllerList{ // LIST
|
|
|
|
Items: []api.ReplicationController{
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: "baz",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1", "k2": "v2"}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
StopError: fmt.Errorf("Detected overlapping controllers for rc foo: baz, please manage deletion individually with --cascade=false."),
|
|
|
|
StopMessage: "",
|
|
|
|
ExpectedActions: []string{"get", "list"},
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Name: "OverlappingButSafeDelete",
|
|
|
|
Objs: []runtime.Object{
|
|
|
|
|
|
|
|
&api.ReplicationController{ // GET
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1", "k2": "v2"}},
|
|
|
|
},
|
|
|
|
&api.ReplicationControllerList{ // LIST
|
|
|
|
Items: []api.ReplicationController{
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: "baz",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1", "k2": "v2", "k3": "v3"}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: "zaz",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1", "k2": "v2"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
StopError: fmt.Errorf("Detected overlapping controllers for rc foo: baz,zaz, please manage deletion individually with --cascade=false."),
|
|
|
|
StopMessage: "",
|
|
|
|
ExpectedActions: []string{"get", "list"},
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
Name: "TwoExactMatchRCs",
|
|
|
|
Objs: []runtime.Object{
|
|
|
|
|
|
|
|
&api.ReplicationController{ // GET
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
&api.ReplicationControllerList{ // LIST
|
|
|
|
Items: []api.ReplicationController{
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: "zaz",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: api.ReplicationControllerSpec{
|
|
|
|
Replicas: 0,
|
|
|
|
Selector: map[string]string{"k1": "v1"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
StopError: nil,
|
|
|
|
StopMessage: "foo stopped",
|
|
|
|
ExpectedActions: []string{"get", "list", "delete"},
|
2015-01-22 17:46:38 +00:00
|
|
|
},
|
|
|
|
}
|
2015-08-12 13:33:08 +00:00
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
fake := testclient.NewSimpleFake(test.Objs...)
|
|
|
|
reaper := ReplicationControllerReaper{fake, time.Millisecond, time.Millisecond}
|
|
|
|
s, err := reaper.Stop(ns, name, 0, nil)
|
|
|
|
if !reflect.DeepEqual(err, test.StopError) {
|
|
|
|
t.Errorf("%s unexpected error: %v", test.Name, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if s != test.StopMessage {
|
|
|
|
t.Errorf("%s expected '%s', got '%s'", test.Name, test.StopMessage, s)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
actions := fake.Actions()
|
|
|
|
if len(actions) != len(test.ExpectedActions) {
|
|
|
|
t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
|
2015-08-03 13:21:11 +00:00
|
|
|
continue
|
|
|
|
}
|
2015-08-12 13:33:08 +00:00
|
|
|
for i, verb := range test.ExpectedActions {
|
|
|
|
if actions[i].GetResource() != "replicationcontrollers" {
|
|
|
|
t.Errorf("%s unexpected action: %+v, expected %s-replicationController", test.Name, actions[i], verb)
|
|
|
|
}
|
|
|
|
if actions[i].GetVerb() != verb {
|
|
|
|
t.Errorf("%s unexpected action: %+v, expected %s-replicationController", test.Name, actions[i], verb)
|
|
|
|
}
|
2015-01-22 17:46:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-16 15:32:59 +00:00
|
|
|
func TestJobStop(t *testing.T) {
|
|
|
|
name := "foo"
|
|
|
|
ns := "default"
|
|
|
|
zero := 0
|
|
|
|
tests := []struct {
|
|
|
|
Name string
|
|
|
|
Objs []runtime.Object
|
|
|
|
StopError error
|
|
|
|
StopMessage string
|
|
|
|
ExpectedActions []string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Name: "OnlyOneJob",
|
|
|
|
Objs: []runtime.Object{
|
2015-10-09 22:49:10 +00:00
|
|
|
&extensions.Job{ // GET
|
2015-09-16 15:32:59 +00:00
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
2015-10-09 22:49:10 +00:00
|
|
|
Spec: extensions.JobSpec{
|
2015-09-16 15:32:59 +00:00
|
|
|
Parallelism: &zero,
|
2015-10-14 18:04:33 +00:00
|
|
|
Selector: &extensions.PodSelector{
|
|
|
|
MatchLabels: map[string]string{"k1": "v1"},
|
|
|
|
},
|
|
|
|
},
|
2015-09-16 15:32:59 +00:00
|
|
|
},
|
2015-10-09 22:49:10 +00:00
|
|
|
&extensions.JobList{ // LIST
|
|
|
|
Items: []extensions.Job{
|
2015-09-16 15:32:59 +00:00
|
|
|
{
|
|
|
|
ObjectMeta: api.ObjectMeta{
|
|
|
|
Name: name,
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
2015-10-09 22:49:10 +00:00
|
|
|
Spec: extensions.JobSpec{
|
2015-09-16 15:32:59 +00:00
|
|
|
Parallelism: &zero,
|
2015-10-14 18:04:33 +00:00
|
|
|
Selector: &extensions.PodSelector{
|
|
|
|
MatchLabels: map[string]string{"k1": "v1"},
|
|
|
|
},
|
|
|
|
},
|
2015-09-16 15:32:59 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
StopError: nil,
|
|
|
|
StopMessage: "foo stopped",
|
|
|
|
ExpectedActions: []string{"get", "get", "update", "get", "get", "delete"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
fake := testclient.NewSimpleFake(test.Objs...)
|
|
|
|
reaper := JobReaper{fake, time.Millisecond, time.Millisecond}
|
|
|
|
s, err := reaper.Stop(ns, name, 0, nil)
|
|
|
|
if !reflect.DeepEqual(err, test.StopError) {
|
|
|
|
t.Errorf("%s unexpected error: %v", test.Name, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if s != test.StopMessage {
|
|
|
|
t.Errorf("%s expected '%s', got '%s'", test.Name, test.StopMessage, s)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
actions := fake.Actions()
|
|
|
|
if len(actions) != len(test.ExpectedActions) {
|
|
|
|
t.Errorf("%s unexpected actions: %v, expected %d actions got %d", test.Name, actions, len(test.ExpectedActions), len(actions))
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for i, verb := range test.ExpectedActions {
|
|
|
|
if actions[i].GetResource() != "jobs" {
|
|
|
|
t.Errorf("%s unexpected action: %+v, expected %s-job", test.Name, actions[i], verb)
|
|
|
|
}
|
|
|
|
if actions[i].GetVerb() != verb {
|
|
|
|
t.Errorf("%s unexpected action: %+v, expected %s-job", test.Name, actions[i], verb)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-22 17:46:38 +00:00
|
|
|
type noSuchPod struct {
|
2015-04-06 23:27:53 +00:00
|
|
|
*testclient.FakePods
|
2015-01-22 17:46:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *noSuchPod) Get(name string) (*api.Pod, error) {
|
|
|
|
return nil, fmt.Errorf("%s does not exist", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
type noDeleteService struct {
|
2015-04-06 23:27:53 +00:00
|
|
|
*testclient.FakeServices
|
2015-01-22 17:46:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *noDeleteService) Delete(service string) error {
|
|
|
|
return fmt.Errorf("I'm afraid I can't do that, Dave")
|
|
|
|
}
|
|
|
|
|
|
|
|
type reaperFake struct {
|
2015-04-06 23:27:53 +00:00
|
|
|
*testclient.Fake
|
2015-01-22 17:46:38 +00:00
|
|
|
noSuchPod, noDeleteService bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *reaperFake) Pods(namespace string) client.PodInterface {
|
2015-08-08 01:52:23 +00:00
|
|
|
pods := &testclient.FakePods{Fake: c.Fake, Namespace: namespace}
|
2015-01-22 17:46:38 +00:00
|
|
|
if c.noSuchPod {
|
|
|
|
return &noSuchPod{pods}
|
|
|
|
}
|
|
|
|
return pods
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *reaperFake) Services(namespace string) client.ServiceInterface {
|
2015-08-08 01:52:23 +00:00
|
|
|
services := &testclient.FakeServices{Fake: c.Fake, Namespace: namespace}
|
2015-01-22 17:46:38 +00:00
|
|
|
if c.noDeleteService {
|
|
|
|
return &noDeleteService{services}
|
|
|
|
}
|
|
|
|
return services
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSimpleStop(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
fake *reaperFake
|
|
|
|
kind string
|
2015-08-03 13:21:11 +00:00
|
|
|
actions []testclient.Action
|
2015-01-22 17:46:38 +00:00
|
|
|
expectError bool
|
|
|
|
test string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
fake: &reaperFake{
|
2015-04-06 23:27:53 +00:00
|
|
|
Fake: &testclient.Fake{},
|
2015-01-22 17:46:38 +00:00
|
|
|
},
|
2015-08-03 13:21:11 +00:00
|
|
|
kind: "Pod",
|
|
|
|
actions: []testclient.Action{
|
|
|
|
testclient.NewGetAction("pods", api.NamespaceDefault, "foo"),
|
|
|
|
testclient.NewDeleteAction("pods", api.NamespaceDefault, "foo"),
|
|
|
|
},
|
2015-01-22 17:46:38 +00:00
|
|
|
expectError: false,
|
|
|
|
test: "stop pod succeeds",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fake: &reaperFake{
|
2015-04-06 23:27:53 +00:00
|
|
|
Fake: &testclient.Fake{},
|
2015-01-22 17:46:38 +00:00
|
|
|
},
|
2015-08-03 13:21:11 +00:00
|
|
|
kind: "Service",
|
|
|
|
actions: []testclient.Action{
|
|
|
|
testclient.NewGetAction("services", api.NamespaceDefault, "foo"),
|
|
|
|
testclient.NewDeleteAction("services", api.NamespaceDefault, "foo"),
|
|
|
|
},
|
2015-01-22 17:46:38 +00:00
|
|
|
expectError: false,
|
|
|
|
test: "stop service succeeds",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fake: &reaperFake{
|
2015-04-06 23:27:53 +00:00
|
|
|
Fake: &testclient.Fake{},
|
2015-01-22 17:46:38 +00:00
|
|
|
noSuchPod: true,
|
|
|
|
},
|
|
|
|
kind: "Pod",
|
2015-08-03 13:21:11 +00:00
|
|
|
actions: []testclient.Action{},
|
2015-01-22 17:46:38 +00:00
|
|
|
expectError: true,
|
|
|
|
test: "stop pod fails, no pod",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fake: &reaperFake{
|
2015-04-06 23:27:53 +00:00
|
|
|
Fake: &testclient.Fake{},
|
2015-01-22 17:46:38 +00:00
|
|
|
noDeleteService: true,
|
|
|
|
},
|
2015-08-03 13:21:11 +00:00
|
|
|
kind: "Service",
|
|
|
|
actions: []testclient.Action{
|
|
|
|
testclient.NewGetAction("services", api.NamespaceDefault, "foo"),
|
|
|
|
},
|
2015-01-22 17:46:38 +00:00
|
|
|
expectError: true,
|
|
|
|
test: "stop service fails, can't delete",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, test := range tests {
|
|
|
|
fake := test.fake
|
2015-08-31 16:29:18 +00:00
|
|
|
reaper, err := ReaperFor(test.kind, fake)
|
2015-01-22 17:46:38 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("unexpected error: %v (%s)", err, test.test)
|
|
|
|
}
|
2015-05-29 23:16:30 +00:00
|
|
|
s, err := reaper.Stop("default", "foo", 0, nil)
|
2015-01-22 17:46:38 +00:00
|
|
|
if err != nil && !test.expectError {
|
|
|
|
t.Errorf("unexpected error: %v (%s)", err, test.test)
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
if test.expectError {
|
|
|
|
t.Errorf("unexpected non-error: %v (%s)", err, test.test)
|
|
|
|
}
|
|
|
|
if s != "foo stopped" {
|
|
|
|
t.Errorf("unexpected return: %s (%s)", s, test.test)
|
|
|
|
}
|
|
|
|
}
|
2015-07-06 21:37:46 +00:00
|
|
|
actions := fake.Actions()
|
|
|
|
if len(test.actions) != len(actions) {
|
2015-01-22 17:46:38 +00:00
|
|
|
t.Errorf("unexpected actions: %v; expected %v (%s)", fake.Actions, test.actions, test.test)
|
|
|
|
}
|
2015-07-06 21:37:46 +00:00
|
|
|
for i, action := range actions {
|
2015-01-22 17:46:38 +00:00
|
|
|
testAction := test.actions[i]
|
2015-08-03 13:21:11 +00:00
|
|
|
if action != testAction {
|
|
|
|
t.Errorf("unexpected action: %#v; expected %v (%s)", action, testAction, test.test)
|
2015-01-22 17:46:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|