Odkryłem, że kody aktualizacji są przechowywane w następującej lokalizacji rejestru.
HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UpgradeCodes
Nazwa klucza rejestru to kod aktualizacji, a nazwa klucza rejestru to kod produktu. Mogę łatwo wyodrębnić te wartości, jednak kody są przechowywane w innym formacie. Czerwone kółko pokazuje sformatowany kod aktualizacji, niebieskie kółko pokazuje sformatowany kod produktu podczas oglądania go w regedit.exe
.

myślników są usuwane z Guid
a następnie seria odwrócenia ciągu są zrobione. Pierwsze 8 znaków jest odwróconych, następne 4, następne 4, a reszta łańcucha jest odwrócona w zestawach po 2 znaki. Zwykle podczas cofania łańcucha musimy zachować ostrożność, upewniając się, że znaki kontrolne i specjalne są obsługiwane poprawnie (see Jon Skeet's aricle here), ale tak jak my, w tym przypadku, traktując ciąg znaków Guid
możemy być pewni, że ciąg znaków zostanie poprawnie odwrócony.
Poniżej znajduje się pełny kod użyty do wyodrębnienia kodu aktualizacji znanego kodu produktu z rejestru.
internal static class RegistryHelper
{
private const string UpgradeCodeRegistryKey = @"SOFTWARE\Microsoft\Windows\CurrentVersion\Installer\UpgradeCodes";
private static readonly int[] GuidRegistryFormatPattern = new[] { 8, 4, 4, 2, 2, 2, 2, 2, 2, 2, 2 };
public static Guid? GetUpgradeCode(Guid productCode)
{
// Convert the product code to the format found in the registry
var productCodeSearchString = ConvertToRegistryFormat(productCode);
// Open the upgrade code registry key
var localMachine = RegistryKey.OpenBaseKey(RegistryHive.LocalMachine, RegistryView.Registry64);
var upgradeCodeRegistryRoot = localMachine.OpenSubKey(UpgradeCodeRegistryKey);
if (upgradeCodeRegistryRoot == null)
return null;
// Iterate over each sub-key
foreach (var subKeyName in upgradeCodeRegistryRoot.GetSubKeyNames())
{
var subkey = upgradeCodeRegistryRoot.OpenSubKey(subKeyName);
if (subkey == null)
continue;
// Check for a value containing the product code
if (subkey.GetValueNames().Any(s => s.IndexOf(productCodeSearchString, StringComparison.OrdinalIgnoreCase) >= 0))
{
// Extract the name of the subkey from the qualified name
var formattedUpgradeCode = subkey.Name.Split('\\').LastOrDefault();
// Convert it back to a Guid
return ConvertFromRegistryFormat(formattedUpgradeCode);
}
}
return null;
}
private static string ConvertToRegistryFormat(Guid productCode)
{
return Reverse(productCode, GuidRegistryFormatPattern);
}
private static Guid ConvertFromRegistryFormat(string upgradeCode)
{
if (upgradeCode == null || upgradeCode.Length != 32)
throw new FormatException("Product code was in an invalid format");
upgradeCode = Reverse(upgradeCode, GuidRegistryFormatPattern);
return Guid.Parse(upgradeCode);
}
private static string Reverse(object value, params int[] pattern)
{
// Strip the hyphens
var inputString = value.ToString().Replace("-", "");
var returnString = new StringBuilder();
var index = 0;
// Iterate over the reversal pattern
foreach (var length in pattern)
{
// Reverse the sub-string and append it
returnString.Append(inputString.Substring(index, length).Reverse().ToArray());
// Increment our posistion in the string
index += length;
}
return returnString.ToString();
}
}
Zamiast korzystania z rejestru zgodnie z sugestiami poniżej ** proponuję użyć WMI, jak opisano w tej odpowiedzi **: [** Jak mogę znaleźć kod Upgrade dla zainstalowanego pliku MSI? **] (https: // stackoverflow.com/questions/46637094/how-can-i-find-the-upgrade-code-for-an-installed-msi-file/46637095 # 46637095). Zapewni to pobranie poprawnego kodu aktualizacji i nie będzie wymagać żadnej konwersji ani interpretacji. ** Otrzymasz prawdziwy kod aktualizacji z powrotem w odpowiednim formacie **. –