All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.spotify.helios.cli.command.PortMappingParser Maven / Gradle / Ivy

/*-
 * -\-\-
 * Helios Tools
 * --
 * Copyright (C) 2016 - 2017 Spotify AB
 * --
 * 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 com.spotify.helios.cli.command;

import static com.google.common.base.Optional.fromNullable;
import static com.spotify.helios.common.descriptors.PortMapping.TCP;
import static java.util.regex.Pattern.compile;

import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Maps;
import com.spotify.helios.common.descriptors.PortMapping;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;


class PortMappingParser {

  private PortMappingParser() {
    // Prevent instantiation
  }

  // TODO (dxia) It'd be great to not have such a specific regex for the "ip" capture group,
  // and let `isInetAddress()` do all the validation. Without the specific regex, however,
  // it's hard to stay backwards compatible. Any ideas on a regex that'll work?
  private static final Pattern PATTERN =
      compile("(?[_\\-\\w]+)=((?([0-9]{1,3}.){3}[0-9]{1,3}):)"
              + "?(?\\d+)(:(?\\d+))?(/(?

\\w+))?"); static PortMappingWithName parsePortMapping(final String portSpec) { final Matcher matcher = PATTERN.matcher(portSpec); if (!matcher.matches()) { throw new IllegalArgumentException("Bad port mapping: " + portSpec); } final String name = matcher.group("n"); final String ip = matcher.group("ip"); final int internal = Integer.parseInt(matcher.group("i")); final Integer external = nullOrInteger(matcher.group("e")); final String protocol = fromNullable(matcher.group("p")).or(TCP); return PortMappingWithName.create(name, PortMapping.builder() .ip(ip) .internalPort(internal) .externalPort(external) .protocol(protocol) .build()); } static Map parsePortMappings(final List portSpecs) { final Map explicitPorts = Maps.newHashMap(); for (final String spec : portSpecs) { final PortMappingWithName portMappingWithName = parsePortMapping(spec); final String name = portMappingWithName.name(); if (explicitPorts.containsKey(name)) { throw new IllegalArgumentException("Duplicate port mapping name: " + name); } explicitPorts.put(name, portMappingWithName.portMapping()); } return ImmutableMap.copyOf(explicitPorts); } /** * A class that simply lets us put the port name and {@link PortMapping} in one object. */ static class PortMappingWithName { private final String name; private final PortMapping portMapping; private PortMappingWithName(final String name, final PortMapping portMapping) { this.name = name; this.portMapping = portMapping; } static PortMappingWithName create(final String name, final PortMapping portMapping) { return new PortMappingWithName(name, portMapping); } String name() { return name; } PortMapping portMapping() { return portMapping; } } private static Integer nullOrInteger(final String str) { return str == null ? null : Integer.valueOf(str); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy