910e62b5创建于 1月15日历史提交
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "device/bluetooth/dbus/bluetooth_profile_manager_client.h"

#include "base/check.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/time/time.h"
#include "dbus/bus.h"
#include "dbus/message.h"
#include "dbus/object_proxy.h"
#include "device/bluetooth/dbus/bluetooth_metrics_helper.h"
#include "third_party/cros_system_api/dbus/service_constants.h"

namespace bluez {

const char BluetoothProfileManagerClient::kNoResponseError[] =
    "org.chromium.Error.NoResponse";

BluetoothProfileManagerClient::Options::Options() = default;

BluetoothProfileManagerClient::Options::~Options() = default;

// The BluetoothProfileManagerClient implementation used in production.
class BluetoothProfileManagerClientImpl : public BluetoothProfileManagerClient {
 public:
  BluetoothProfileManagerClientImpl() {}

  BluetoothProfileManagerClientImpl(const BluetoothProfileManagerClientImpl&) =
      delete;
  BluetoothProfileManagerClientImpl& operator=(
      const BluetoothProfileManagerClientImpl&) = delete;

  ~BluetoothProfileManagerClientImpl() override = default;

  // BluetoothProfileManagerClient override.
  void RegisterProfile(const dbus::ObjectPath& profile_path,
                       const std::string& uuid,
                       const Options& options,
                       base::OnceClosure callback,
                       ErrorCallback error_callback) override {
    dbus::MethodCall method_call(
        bluetooth_profile_manager::kBluetoothProfileManagerInterface,
        bluetooth_profile_manager::kRegisterProfile);

    dbus::MessageWriter writer(&method_call);
    writer.AppendObjectPath(profile_path);
    writer.AppendString(uuid);

    dbus::MessageWriter array_writer(nullptr);
    writer.OpenArray("{sv}", &array_writer);

    dbus::MessageWriter main_dict_writer(nullptr);

    // Send Name if provided.
    if (options.name.get() != nullptr) {
      array_writer.OpenDictEntry(&main_dict_writer);
      main_dict_writer.AppendString(bluetooth_profile_manager::kNameOption);
      main_dict_writer.AppendVariantOfString(*(options.name));
      array_writer.CloseContainer(&main_dict_writer);
    }

    // Send Service if provided.
    if (options.service.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kServiceOption);
      dict_writer.AppendVariantOfString(*(options.service));
      array_writer.CloseContainer(&dict_writer);
    }

    // Send Role if not the default value.
    if (options.role != SYMMETRIC) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kRoleOption);
      if (options.role == CLIENT)
        dict_writer.AppendVariantOfString(
            bluetooth_profile_manager::kClientRoleOption);
      else if (options.role == SERVER)
        dict_writer.AppendVariantOfString(
            bluetooth_profile_manager::kServerRoleOption);
      else
        dict_writer.AppendVariantOfString("");
      array_writer.CloseContainer(&dict_writer);
    }

    // Send Channel if provided.
    if (options.channel.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kChannelOption);
      dict_writer.AppendVariantOfUint16(*(options.channel));
      array_writer.CloseContainer(&dict_writer);
    }

    // Send PSM if provided.
    if (options.psm.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kPSMOption);
      dict_writer.AppendVariantOfUint16(*(options.psm));
      array_writer.CloseContainer(&dict_writer);
    }

    // Send RequireAuthentication if provided.
    if (options.require_authentication.get() != nullptr) {
      array_writer.OpenDictEntry(&main_dict_writer);
      main_dict_writer.AppendString(
          bluetooth_profile_manager::kRequireAuthenticationOption);
      main_dict_writer.AppendVariantOfBool(*(options.require_authentication));
      array_writer.CloseContainer(&main_dict_writer);
    }

    // Send RequireAuthorization if provided.
    if (options.require_authorization.get() != nullptr) {
      array_writer.OpenDictEntry(&main_dict_writer);
      main_dict_writer.AppendString(
          bluetooth_profile_manager::kRequireAuthorizationOption);
      main_dict_writer.AppendVariantOfBool(*(options.require_authorization));
      array_writer.CloseContainer(&main_dict_writer);
    }

    // Send AutoConnect if provided.
    if (options.auto_connect.get() != nullptr) {
      array_writer.OpenDictEntry(&main_dict_writer);
      main_dict_writer.AppendString(
          bluetooth_profile_manager::kAutoConnectOption);
      main_dict_writer.AppendVariantOfBool(*(options.auto_connect));
      array_writer.CloseContainer(&main_dict_writer);
    }

    // Send ServiceRecord if provided.
    if (options.service_record.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kServiceRecordOption);
      dict_writer.AppendVariantOfString(*(options.service_record));
      array_writer.CloseContainer(&dict_writer);
    }

    // Send Version if provided.
    if (options.version.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kVersionOption);
      dict_writer.AppendVariantOfUint16(*(options.version));
      array_writer.CloseContainer(&dict_writer);
    }

    // Send Features if provided.
    if (options.features.get() != nullptr) {
      dbus::MessageWriter dict_writer(nullptr);
      array_writer.OpenDictEntry(&dict_writer);
      dict_writer.AppendString(bluetooth_profile_manager::kFeaturesOption);
      dict_writer.AppendVariantOfUint16(*(options.features));
      array_writer.CloseContainer(&dict_writer);
    }

    writer.CloseContainer(&array_writer);

    object_proxy_->CallMethodWithErrorResponse(
        &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT,
        base::BindOnce(&BluetoothProfileManagerClientImpl::OnRegisterProfile,
                       weak_ptr_factory_.GetWeakPtr(), std::move(callback),
                       std::move(error_callback), base::Time::Now()));
  }

  // BluetoothProfileManagerClient override.
  void UnregisterProfile(const dbus::ObjectPath& profile_path,
                         base::OnceClosure callback,
                         ErrorCallback error_callback) override {
    dbus::MethodCall method_call(
        bluetooth_profile_manager::kBluetoothProfileManagerInterface,
        bluetooth_profile_manager::kUnregisterProfile);

    dbus::MessageWriter writer(&method_call);
    writer.AppendObjectPath(profile_path);

    object_proxy_->CallMethodWithErrorResponse(
        &method_call, dbus::ObjectProxy::TIMEOUT_USE_DEFAULT,
        base::BindOnce(&BluetoothProfileManagerClientImpl::OnUnregisterProfile,
                       weak_ptr_factory_.GetWeakPtr(), std::move(callback),
                       std::move(error_callback), base::Time::Now()));
  }

 protected:
  void Init(dbus::Bus* bus,
            const std::string& bluetooth_service_name) override {
    DCHECK(bus);
    object_proxy_ = bus->GetObjectProxy(
        bluetooth_service_name,
        dbus::ObjectPath(
            bluetooth_profile_manager::kBluetoothProfileManagerServicePath));
  }

 private:
  void OnRegisterProfile(base::OnceClosure callback,
                         ErrorCallback error_callback,
                         base::Time start_time,
                         dbus::Response* response,
                         dbus::ErrorResponse* error_response) {
    if (response) {
      RecordSuccess(kRegisterProfileMethod, start_time);
      std::move(callback).Run();
    } else {
      RecordFailure(kRegisterProfileMethod, error_response);
      OnError(std::move(error_callback), error_response);
    }
  }

  void OnUnregisterProfile(base::OnceClosure callback,
                           ErrorCallback error_callback,
                           base::Time start_time,
                           dbus::Response* response,
                           dbus::ErrorResponse* error_response) {
    if (response) {
      RecordSuccess(kUnregisterProfileMethod, start_time);
      std::move(callback).Run();
    } else {
      RecordFailure(kUnregisterProfileMethod, error_response);
      OnError(std::move(error_callback), error_response);
    }
  }

  // Called when a response for a failed method call is received.
  void OnError(ErrorCallback error_callback, dbus::ErrorResponse* response) {
    // Error response has optional error message argument.
    std::string error_name;
    std::string error_message;
    if (response) {
      dbus::MessageReader reader(response);
      error_name = response->GetErrorName();
      reader.PopString(&error_message);
    } else {
      error_name = kNoResponseError;
      error_message = "";
    }
    std::move(error_callback).Run(error_name, error_message);
  }

  raw_ptr<dbus::ObjectProxy> object_proxy_;

  // Weak pointer factory for generating 'this' pointers that might live longer
  // than we do.
  // Note: This should remain the last member so it'll be destroyed and
  // invalidate its weak pointers before any other members are destroyed.
  base::WeakPtrFactory<BluetoothProfileManagerClientImpl> weak_ptr_factory_{
      this};
};

BluetoothProfileManagerClient::BluetoothProfileManagerClient() = default;

BluetoothProfileManagerClient::~BluetoothProfileManagerClient() = default;

BluetoothProfileManagerClient* BluetoothProfileManagerClient::Create() {
  return new BluetoothProfileManagerClientImpl();
}

}  // namespace bluez