# Copyright 2017 Cloudbase Solutions Srl
#
#    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.

from oslo_log import log as oslo_logging
import wmi

from cloudbaseinit import constant
from cloudbaseinit import exception
from cloudbaseinit.osutils import factory as osutils_factory

LOG = oslo_logging.getLogger(__name__)


STORE_CURRENT = "{fa926493-6f1c-4193-a414-58f0b2456d1e}"


BCDOSLOADER_DEVICE_OSDEVICE = 0x21000001
BCDLIBRARY_DEVICE_APPLICATION_DEVICE = 0x11000001
BCDLIBRARY_BOOLEAN_AUTO_RECOVERY_ENABLED = 0x16000009
BOOT_DEVICE = 1


def _run_bcdedit(bcdedit_args):
    args = ["bcdedit.exe"] + bcdedit_args
    osutils = osutils_factory.get_os_utils()
    (out, err, ret_val) = osutils.execute_system32_process(args)
    if ret_val:
        raise exception.CloudbaseInitException(
            'bcdedit failed.\nOutput: %(out)s\nError:'
            ' %(err)s' % {'out': out, 'err': err})


def set_boot_status_policy(policy=constant.POLICY_IGNORE_ALL_FAILURES):
    LOG.debug("Setting boot status policy: %s", policy)
    _run_bcdedit(["/set", "{current}", "bootstatuspolicy", policy])


def get_boot_system_devices():
    conn = wmi.WMI(moniker='//./root/cimv2')
    return [v.DeviceID for v in conn.Win32_Volume(
        BootVolume=True, SystemVolume=True)]


def _get_current_bcd_store():
    conn = wmi.WMI(moniker='//./root/wmi')
    success, store = conn.BcdStore.OpenStore(File="")
    if not success:
        raise exception.CloudbaseInitException("Cannot open BCD store")
    store = wmi._wmi_object(store)
    current_store, success = store.OpenObject(Id=STORE_CURRENT)
    current_store = wmi._wmi_object(current_store)
    if not success:
        raise exception.CloudbaseInitException("Cannot open BCD current store")

    return current_store


def set_current_bcd_device_to_boot_partition():
    current_store = _get_current_bcd_store()

    success, = current_store.SetDeviceElement(
        Type=BCDOSLOADER_DEVICE_OSDEVICE, DeviceType=BOOT_DEVICE,
        AdditionalOptions="")
    if not success:
        raise exception.CloudbaseInitException(
            "Cannot set device element: %s" % BCDOSLOADER_DEVICE_OSDEVICE)

    success, = current_store.SetDeviceElement(
        Type=BCDLIBRARY_DEVICE_APPLICATION_DEVICE, DeviceType=BOOT_DEVICE,
        AdditionalOptions="")
    if not success:
        raise exception.CloudbaseInitException(
            "Cannot set device element: %s" %
            BCDLIBRARY_DEVICE_APPLICATION_DEVICE)


def enable_auto_recovery(enable):
    current_store = _get_current_bcd_store()

    success, = current_store.SetBooleanElement(
        BCDLIBRARY_BOOLEAN_AUTO_RECOVERY_ENABLED, enable)
    if not success:
        raise exception.CloudbaseInitException(
            "Cannot set boolean element: %s" %
            BCDLIBRARY_BOOLEAN_AUTO_RECOVERY_ENABLED)