Настройка внутренних сетей Ethernet

Android Auto OS 13 и более поздних версий содержит функции, позволяющие настраивать сети Ethernet и управлять ими. На рисунке 1 показан пример сетевой диаграммы автомобиля:

Сеть Android Auto

Рисунок 1. Сеть Android Auto.

На этом рисунке показаны методы вызова вашего OEM-сетевого приложения в классе EthernetManager для настройки и управления встроенными сетями Ethernet (eth0.1, eth0.2 и eth0.3). Оставшаяся часть рисунка 1 выходит за рамки данного документа.

Установите настройки сети Ethernet по умолчанию

Чтобы установить настройки сети по умолчанию, используйте наложение ресурса config_ethernet_interfaces :

<string-array translatable="false" name="config_ethernet_interfaces">
        <!--
        <item>eth1;12,13,14,15;ip=192.168.0.10/24 gateway=192.168.0.1 dns=4.4.4.4,8.8.8.8</item>
        <item>eth2;;ip=192.168.0.11/24</item>
        <item>eth3;12,13,14,15;ip=192.168.0.12/24;1</item>
        -->
    </string-array>

В этом примере показано наложение ресурса config_ethernet_interfaces из config.xml .

Ключевые моменты о коде

  • eth1 , eth2 и eth3 — это имена настраиваемого сетевого интерфейса.
  • Последовательные цифры 12, 13, 14, 15 обозначают включенные сетевые возможности .
  • ip= , gateway= и dns используются для установки начального IP-адреса, шлюза и DNS для сети.

Включить или отключить сетевой интерфейс

Чтобы включить сетевой интерфейс, вызовите EthernetManager.enableInterface() :

public final class InterfaceEnabler {
    private final Context mApplicationContext;
    private final EthernetManager mEthernetManager;
    private final OutcomeReceiver<String, EthernetNetworkManagementException> mOutcomeReceiver;

    public InterfaceEnabler(Context applicationContext,
            OutcomeReceiver<String, EthernetNetworkManagementException> outcomeReceiver) {
        mApplicationContext = applicationContext;
        mEthernetManager = applicationContext.getSystemService(EthernetManager.class);
        mOutcomeReceiver = outcomeReceiver;
    }

    public void enableInterface(String ifaceName) {
        mEthernetManager.enableInterface(ifaceName,
                mApplicationContext.getMainExecutor(),
                mOutcomeReceiver);
    }
}

Ключевые моменты о коде

  • ifaceName — имя сетевого интерфейса, который необходимо включить.
  • getMainExecutor() возвращает контекст приложения.
  • OutcomeReceiver — это обратный вызов, используемый для сообщения о завершении, возвращающий обновленное имя сети в случае успеха или EthernetNetworkManagementException в случае ошибки.

Когда сетевой интерфейс включен, он использует конфигурацию, установленную EthernetManager.updateConfiguration() . Если конфигурация не была установлена ​​с помощью EthernetManager.updateConfiguration() , сетевой интерфейс использует оверлей ресурса config_ethernet_interfaces или конфигурацию сети Ethernet по умолчанию, если оверлей недоступен.

Чтобы отключить сетевой интерфейс, вызовите EthernetManager.disableInterface() :

public final class InterfaceEnabler {
    private final Context mApplicationContext;
    private final EthernetManager mEthernetManager;
    private final OutcomeReceiver<String, EthernetNetworkManagementException> mOutcomeReceiver;

    public InterfaceEnabler(Context applicationContext,
            OutcomeReceiver<String, EthernetNetworkManagementException> outcomeReceiver) {
        mApplicationContext = applicationContext;
        mEthernetManager = applicationContext.getSystemService(EthernetManager.class);
        mOutcomeReceiver = outcomeReceiver;
    }

    public void disableInterface(String ifaceName) {
        mEthernetManager.disableInterface(ifaceName,
                mApplicationContext.getMainExecutor(),
                mOutcomeReceiver);
    }
}

Ключевые моменты о коде

  • ifaceName — имя сетевого интерфейса, который нужно отключить.
  • getMainExecutor() возвращает контекст приложения.
  • OutcomeReceiver — это обратный вызов, используемый для сообщения о завершении, возвращающий обновленное имя сети в случае успеха или EthernetNetworkManagementException в случае ошибки.

Обновить конфигурацию сети

Чтобы обновить конфигурации сети Ethernet, вызовите EthernetManager.updateConfiguration() :

public final class ConfigurationUpdater {
    private final Context mApplicationContext;
    private final EthernetManager mEthernetManager;
    private final OutcomeReceiver<String, EthernetNetworkManagementException> mCallback;

    public ConfigurationUpdater(Context applicationContext,
            OutcomeReceiver<String, EthernetNetworkManagementException> callback) {
        mApplicationContext = applicationContext;
        mEthernetManager = applicationContext.getSystemService(EthernetManager.class);
        mCallback = callback;
    }

    public void updateNetworkConfiguration(String packageNames,
            String ipConfigurationText,
            String networkCapabilitiesText,
            String interfaceName)
            throws IllegalArgumentException, PackageManager.NameNotFoundException {

        EthernetNetworkUpdateRequest request = new EthernetNetworkUpdateRequest.Builder()
                .setIpConfiguration(getIpConfiguration(ipConfigurationText))
                .setNetworkCapabilities(getCapabilities(
                        interfaceName, networkCapabilitiesText, packageNames))
                .build();

        mEthernetManager.updateConfiguration(interfaceName, request,
                mApplicationContext.getMainExecutor(), mCallback);

    }
}

Ключевые моменты о коде

  • getCapabilities() — это вспомогательный метод, который получает текущие сетевые возможности и вызывает convertToUIDs() для преобразования удобочитаемых имен пакетов в уникальный идентификатор Linux (UID) . Обычно вы заранее не знаете UID для связанных с ними пакетов. Поэтому, если вы хотите использовать EthernetManager.updateConfiguration() для ограничения доступа к подмножеству приложений, вам необходимо использовать их UID.
  • request — это конфигурация, которая будет использоваться для внутренней сети. Запрос может содержать новые настройки конфигурации IP и возможностей сети. Если сеть зарегистрирована в стеке подключений, она обновляется в соответствии с конфигурацией. Эта конфигурация не сохраняется при перезагрузках.
  • getMainExecutor() возвращает исполнителя, у которого вызывается прослушиватель.
  • mCallback — это обратный вызов, используемый для сообщения о завершении, возвращающий обновленное имя сети в случае успеха или EthernetNetworkManagementException в случае ошибки.

updateConfiguration() может обновить характеристики сети, которую стек Android Connectivity считает неизменной. Сеть отключается, обновляется и снова восстанавливается для обновления этих неизменяемых атрибутов.

Ограничить сеть подмножеством приложений

Вы можете использовать EthernetManager#updateConfiguration чтобы ограничить доступ только к подмножеству разрешенных UID. Используйте этот метод, чтобы охватить случаи, когда это необходимо, например, для внутренних транспортных сетей, которые могут использоваться только небольшим подмножеством OEM-приложений.

Android в первую очередь отслеживает приложения по их UID. Следующий код из UIDToPackageNameConverter.java показывает, как получить серию UID из строки имен пакетов:

public static Set<Integer> convertToUids(Context applicationContext, String packageNames)
            throws PackageManager.NameNotFoundException {
        final PackageManager packageManager = applicationContext.getPackageManager();
        final UserManager userManager = applicationContext.getSystemService(UserManager.class);

        final Set<Integer> uids = new ArraySet<>();
        final List<UserHandle> users = userManager.getUserHandles(true);

        String[] packageNamesArray = packageNames.split(",");
        for (String packageName : packageNamesArray) {
            boolean nameNotFound = true;
            packageName = packageName.trim();
            for (final UserHandle user : users) {
                try {
                    final int uid =
                            packageManager.getApplicationInfoAsUser(packageName, 0, user).uid;
                    uids.add(uid);
                    nameNotFound = false;
                } catch (PackageManager.NameNotFoundException e) {
                    // Although this may seem like an error scenario, it is OK as all packages are
                    // not expected to be installed for all users.
                    continue;
                }
            }

            if (nameNotFound) {
                throw new PackageManager.NameNotFoundException("Not installed: " + packageName);
            }
        }
        return uids;

Ключевые моменты о коде

  • getApplicationInfoAsuser().uid используется для получения UID из имени пакета.
  • uids — это сгенерированный массив целых чисел.

Следующий код в EthernetManagerTest.kt показывает, как обновить конфигурацию сетевого интерфейса с помощью UID приложений, которым разрешено использовать сеть:

val allowedUids = setOf(Process.myUid())
        val nc = NetworkCapabilities.Builder(request.networkCapabilities)
                .setAllowedUids(allowedUids).build()
        updateConfiguration(iface, capabilities = nc).expectResult(iface.name)

Где:

  • allowUids — это набор UID приложений, которым разрешено использовать сеть.
  • updateConfiguration() обновляет конфигурацию, ограничивая сеть предоставленным набором UID.