| /* |
| * Copyright (C) 2022 The Android Open Source Project |
| * |
| * 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.android.car.telemetry.publisher; |
| |
| import android.annotation.NonNull; |
| import android.app.StatsManager; |
| import android.content.Context; |
| import android.net.INetworkStatsService; |
| import android.os.Handler; |
| import android.os.IBinder; |
| import android.os.ServiceManager; |
| |
| import com.android.car.CarPropertyService; |
| import com.android.car.telemetry.TelemetryProto; |
| import com.android.car.telemetry.publisher.net.NetworkStatsServiceProxy; |
| import com.android.internal.util.Preconditions; |
| |
| import java.io.File; |
| import java.util.Objects; |
| |
| /** |
| * Lazy factory class for Publishers. It's expected to have a single factory instance. Must be |
| * called from the telemetry thread. |
| * |
| * <p>It doesn't instantiate all the publishers right away, as in some cases some publishers are not |
| * needed. |
| * |
| * <p>Methods in this class must be called on telemetry thread unless specified as thread-safe. |
| */ |
| public class PublisherFactory { |
| // Some publishers must be initialized as early as possible during boot. |
| private static final TelemetryProto.Publisher.PublisherCase[] sForceInitPublishers = { |
| TelemetryProto.Publisher.PublisherCase.CONNECTIVITY |
| }; |
| |
| private final Object mLock = new Object(); |
| private final CarPropertyService mCarPropertyService; |
| private final File mPublisherDirectory; |
| private final Handler mTelemetryHandler; |
| private final Context mContext; // CarService context |
| |
| private VehiclePropertyPublisher mVehiclePropertyPublisher; |
| private CarTelemetrydPublisher mCarTelemetrydPublisher; |
| private StatsPublisher mStatsPublisher; |
| private ConnectivityPublisher mConnectivityPublisher; |
| private AbstractPublisher.PublisherFailureListener mFailureListener; |
| |
| public PublisherFactory( |
| @NonNull CarPropertyService carPropertyService, |
| @NonNull Handler handler, |
| @NonNull Context context, |
| @NonNull File publisherDirectory) { |
| mCarPropertyService = carPropertyService; |
| mTelemetryHandler = handler; |
| mContext = context; |
| mPublisherDirectory = publisherDirectory; |
| } |
| |
| /** Returns the publisher by given type. This method is thread-safe. */ |
| @NonNull |
| public AbstractPublisher getPublisher(@NonNull TelemetryProto.Publisher.PublisherCase type) { |
| Preconditions.checkState(mFailureListener != null, "PublisherFactory is not initialized"); |
| // No need to optimize locks, as this method is infrequently called. |
| synchronized (mLock) { |
| switch (type.getNumber()) { |
| case TelemetryProto.Publisher.VEHICLE_PROPERTY_FIELD_NUMBER: |
| if (mVehiclePropertyPublisher == null) { |
| mVehiclePropertyPublisher = new VehiclePropertyPublisher( |
| mCarPropertyService, mFailureListener, mTelemetryHandler); |
| } |
| return mVehiclePropertyPublisher; |
| case TelemetryProto.Publisher.CARTELEMETRYD_FIELD_NUMBER: |
| if (mCarTelemetrydPublisher == null) { |
| mCarTelemetrydPublisher = new CarTelemetrydPublisher( |
| mFailureListener, mTelemetryHandler); |
| } |
| return mCarTelemetrydPublisher; |
| case TelemetryProto.Publisher.STATS_FIELD_NUMBER: |
| if (mStatsPublisher == null) { |
| StatsManager stats = mContext.getSystemService(StatsManager.class); |
| Preconditions.checkState(stats != null, "StatsManager not found"); |
| StatsManagerProxy statsManager = new StatsManagerImpl(stats); |
| mStatsPublisher = new StatsPublisher( |
| mFailureListener, statsManager, mPublisherDirectory, |
| mTelemetryHandler); |
| } |
| return mStatsPublisher; |
| case TelemetryProto.Publisher.CONNECTIVITY_FIELD_NUMBER: |
| if (mConnectivityPublisher == null) { |
| IBinder service = |
| Objects.requireNonNull( |
| ServiceManager.getService(Context.NETWORK_STATS_SERVICE)); |
| INetworkStatsService networkStatsService = |
| INetworkStatsService.Stub.asInterface(service); |
| mConnectivityPublisher = |
| new ConnectivityPublisher( |
| mFailureListener, |
| new NetworkStatsServiceProxy(networkStatsService), |
| mTelemetryHandler, |
| mContext); |
| } |
| return mConnectivityPublisher; |
| default: |
| throw new IllegalArgumentException( |
| "Publisher type " + type + " is not supported"); |
| } |
| } |
| } |
| |
| /** |
| * Removes all {@link com.android.car.telemetry.databroker.DataSubscriber} from all publishers. |
| */ |
| public void removeAllDataSubscribers() { |
| if (mVehiclePropertyPublisher != null) { |
| mVehiclePropertyPublisher.removeAllDataSubscribers(); |
| } |
| if (mCarTelemetrydPublisher != null) { |
| mCarTelemetrydPublisher.removeAllDataSubscribers(); |
| } |
| if (mStatsPublisher != null) { |
| mStatsPublisher.removeAllDataSubscribers(); |
| } |
| } |
| |
| /** |
| * Initializes the factory and sets the publisher failure listener for all the publishers. |
| * This is expected to be called before {@link #getPublisher} method. This is not the best |
| * approach, but it suits for this case. |
| */ |
| public void initialize(@NonNull AbstractPublisher.PublisherFailureListener listener) { |
| Preconditions.checkState( |
| mFailureListener == null, "PublisherFactory is already initialized"); |
| mFailureListener = listener; |
| for (TelemetryProto.Publisher.PublisherCase publisher: sForceInitPublishers) { |
| getPublisher(publisher); |
| } |
| } |
| } |