chromium/chrome/browser/ash/ownership/owner_key_loader.h

// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef CHROME_BROWSER_ASH_OWNERSHIP_OWNER_KEY_LOADER_H_
#define CHROME_BROWSER_ASH_OWNERSHIP_OWNER_KEY_LOADER_H_

#include "base/feature_list.h"
#include "base/functional/callback.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/ref_counted.h"
#include "base/memory/weak_ptr.h"
#include "components/ownership/owner_key_util.h"

class Profile;

namespace enterprise_management {
class PolicyData;
}

namespace ash {

class DeviceSettingsService;

BASE_DECLARE_FEATURE(kStoreOwnerKeyInPrivateSlot);
BASE_DECLARE_FEATURE(kMigrateOwnerKeyToPrivateSlot);

// A helper single-use class to load the owner key.
// Determines whether the current user is the owner or not.
// For the non-owner just loads the public owner key (which can be used to
// verify signature on the device policies).
// For the owner loads both public and private key or generates new ones if the
// previous ones were lost.
// For the first user that should become the owner generates a new key pair.
// All public methods might depend on the profile and therefore should be run on
// the UI thread.
class OwnerKeyLoader {
 public:
  using KeypairCallback = base::OnceCallback<void(
      scoped_refptr<ownership::PublicKey> public_key,
      scoped_refptr<ownership::PrivateKey> private_key)>;

  OwnerKeyLoader(Profile* profile,
                 DeviceSettingsService* device_settings_service,
                 scoped_refptr<ownership::OwnerKeyUtil> owner_key_util,
                 bool is_enterprise_managed,
                 KeypairCallback callback);
  OwnerKeyLoader(const OwnerKeyLoader&) = delete;
  auto operator=(const OwnerKeyLoader&) = delete;
  ~OwnerKeyLoader();

  // Starts the loading of the key(s). Can be called only once per instance of
  // the class.
  void Run();

  // Extracts the old owner key when it was replaced. Should be called after
  // Run() and after the `callback` (from the constructor) is called. Returns a
  // nullptr if the key was not replaced.
  // TODO(b/264397430): The method can be removed after the migration is done.
  crypto::ScopedSECKEYPrivateKey ExtractOldOwnerKey();

 private:
  void OnPublicKeyLoaded(scoped_refptr<ownership::PublicKey> public_key);
  void OnPrivateKeyLoaded(scoped_refptr<ownership::PrivateKey> private_key,
                          bool found_in_public_slot);
  void MaybeGenerateNewKey();
  void GenerateNewKey();
  void OnNewKeyGenerated(scoped_refptr<ownership::PublicKey> public_key,
                         scoped_refptr<ownership::PrivateKey> private_key);
  void MaybeRegenerateLostKey(
      const enterprise_management::PolicyData* policy_data);

  const raw_ptr<Profile> profile_;
  const raw_ptr<DeviceSettingsService> device_settings_service_;
  scoped_refptr<ownership::OwnerKeyUtil> owner_key_util_;
  const bool is_enterprise_managed_;
  scoped_refptr<ownership::PublicKey> public_key_;
  KeypairCallback callback_;
  int generate_attempt_counter_ = 0;
  // If not null, contains the old owner key that is currently being replaced by
  // the MigrateOwnerKeyToPrivateSlot experiment.
  // TODO(b/264397430): This can be removed after the migration is finished.
  crypto::ScopedSECKEYPrivateKey old_owner_key_;

  base::WeakPtrFactory<OwnerKeyLoader> weak_factory_{this};
};

}  // namespace ash

#endif  // CHROME_BROWSER_ASH_OWNERSHIP_OWNER_KEY_LOADER_H_