summaryrefslogtreecommitdiffstats
path: root/nsh/impl/src/main/java/io/fd/honeycomb/vppnsh/impl/config/NshMapWriterCustomizer.java
blob: f2757bd64f22e669f2946fc048677994a275c1c6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
/*
 * Copyright (c) 2016 Intel and/or its affiliates.
 *
 * 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 io.fd.honeycomb.vppnsh.impl.config;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;
import static io.fd.honeycomb.translate.v3po.util.TranslateUtils.booleanToByte;

import io.fd.honeycomb.translate.spi.write.ListWriterCustomizer;
import io.fd.honeycomb.translate.v3po.util.NamingContext;
import io.fd.honeycomb.translate.v3po.util.WriteTimeoutException;
import io.fd.honeycomb.translate.v3po.util.TranslateUtils;
import io.fd.honeycomb.translate.write.WriteContext;
import io.fd.honeycomb.translate.write.WriteFailedException;
import io.fd.honeycomb.translate.MappingContext;
import java.util.concurrent.CompletionStage;
import javax.annotation.Nonnull;
import javax.xml.bind.DatatypeConverter;
import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.vpp.nsh.rev160624.EncapType;
import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.vpp.nsh.rev160624.VxlanGpe;
import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.vpp.nsh.rev160624.vpp.nsh.nsh.maps.NshMap;
import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.vpp.nsh.rev160624.vpp.nsh.nsh.maps.NshMapKey;
import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
import org.openvpp.jvpp.VppBaseCallException;
import org.openvpp.jvpp.nsh.dto.*;
import org.openvpp.jvpp.nsh.callback.*;
import org.openvpp.jvpp.nsh.future.FutureJVppNsh;
import io.fd.honeycomb.vppnsh.impl.util.FutureJVppNshCustomizer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Writer customizer responsible for NshMap create/delete.
 */
public class NshMapWriterCustomizer extends FutureJVppNshCustomizer
    implements ListWriterCustomizer<NshMap, NshMapKey> {

    private static final Logger LOG = LoggerFactory.getLogger(NshMapWriterCustomizer.class);
    private final NamingContext nshMapContext;
    private final NamingContext interfaceContext;

    public NshMapWriterCustomizer(@Nonnull final FutureJVppNsh futureJVppNsh,
                                    @Nonnull final NamingContext nshMapContext,
                                    @Nonnull final NamingContext interfaceContext) {
        super(futureJVppNsh);
        this.nshMapContext = checkNotNull(nshMapContext, "nshMapContext should not be null");
        this.interfaceContext = checkNotNull(interfaceContext, "interfaceContext should not be null");
    }

    @Override
    public void writeCurrentAttributes(@Nonnull final InstanceIdentifier<NshMap> id,
                                       @Nonnull final NshMap dataAfter, @Nonnull final WriteContext writeContext)
        throws WriteFailedException {
        LOG.debug("Creating nsh map: iid={} dataAfter={}", id, dataAfter);
        try {
            final int newMapIndex =
                nshAddDelMap(true, id, dataAfter, ~0 /* value not present */, writeContext.getMappingContext());

            // Add nsh map name <-> vpp index mapping to the naming context:
            nshMapContext.addName(newMapIndex, dataAfter.getName(), writeContext.getMappingContext());
            LOG.debug("Successfully created nsh map(id={]): iid={} dataAfter={}", newMapIndex, id, dataAfter);
        } catch (VppBaseCallException e) {
            throw new WriteFailedException.CreateFailedException(id, dataAfter, e);
        }
    }

    @Override
    public void updateCurrentAttributes(@Nonnull final InstanceIdentifier<NshMap> id,
                                        @Nonnull final NshMap dataBefore, @Nonnull final NshMap dataAfter,
                                        @Nonnull final WriteContext writeContext) throws WriteFailedException {
        throw new UnsupportedOperationException("Nsh map update is not supported");
    }

    @Override
    public void deleteCurrentAttributes(@Nonnull final InstanceIdentifier<NshMap> id,
                                        @Nonnull final NshMap dataBefore,
                                        @Nonnull final WriteContext writeContext) throws WriteFailedException {
        LOG.debug("Removing nsh map: iid={} dataBefore={}", id, dataBefore);
        final String mapName = dataBefore.getName();
        checkState(nshMapContext.containsIndex(mapName, writeContext.getMappingContext()),
            "Removing nsh map {}, but index could not be found in the nsh map context", mapName);

        final int mapIndex = nshMapContext.getIndex(mapName, writeContext.getMappingContext());
        try {
            nshAddDelMap(false, id, dataBefore, mapIndex, writeContext.getMappingContext());

            // Remove deleted interface from interface context:
            nshMapContext.removeName(dataBefore.getName(), writeContext.getMappingContext());
            LOG.debug("Successfully removed nsh map(id={]): iid={} dataAfter={}", mapIndex, id, dataBefore);
        } catch (VppBaseCallException e) {
            throw new WriteFailedException.DeleteFailedException(id, e);
        }
    }

    private int nshAddDelMap(final boolean isAdd, @Nonnull final InstanceIdentifier<NshMap> id,
                                    @Nonnull final NshMap map, final int mapId, final MappingContext ctx)
        throws VppBaseCallException, WriteTimeoutException {
        final CompletionStage<NshAddDelMapReply> createNshMapReplyCompletionStage =
            getFutureJVppNsh().nshAddDelMap(getNshAddDelMapRequest(isAdd, mapId, map, ctx));

        final NshAddDelMapReply reply =
            TranslateUtils.getReplyForWrite(createNshMapReplyCompletionStage.toCompletableFuture(), id);
        return reply.mapIndex;

    }

    private NshAddDelMap getNshAddDelMapRequest(final boolean isAdd, final int mapIndex,
                                                    @Nonnull final NshMap map,
                                                    @Nonnull final MappingContext ctx) {
        final NshAddDelMap request = new NshAddDelMap();
        request.isAdd = booleanToByte(isAdd);

        request.nspNsi = (map.getNsp().intValue()<<8) | map.getNsi();
        request.mappedNspNsi = (map.getMappedNsp().intValue()<<8) | map.getMappedNsi();

        if (map.getEncapType() == VxlanGpe.class) {
            request.nextNode = 2;
        }

        checkState(interfaceContext.containsIndex(map.getEncapIfName(), ctx),
                "Mapping does not contains mapping for provider interface Name ".concat(map.getEncapIfName()));
        request.swIfIndex = interfaceContext.getIndex(map.getEncapIfName(), ctx);

        return request;
    }
}