Unikalny numer komputera

Unikalny numer komputera
Kaska1988
  • Rejestracja:prawie 12 lat
  • Ostatnio:5 miesięcy
  • Postów:186
0

Witam,

czy ktoś z Was wie może jak wygenerować unikalny numer maszyny? Próbowałam z wygenerowaniem numeru seryjnego dysku twardego, który znalazłam na torrys.net ale coś mi to nie wychodziło. Pytam bo potrzebne mi to do rozróżniania komputerów w sieci a kombinacja adres ip + nazwa użytkownika się nie sprawdza (jak zapewne wiecie jest szansa, że trafią się komputery z adresem IP 192.168.0.1 i nazwą użytkownika Administrator).

Pozdrawiam!

PS. Nie chodzi mi o numer aktualnej sesji bo to nie rozwiązuje problemu.


_13th_Dragon
  • Rejestracja:ponad 19 lat
  • Ostatnio:około 18 godzin
1

użyj MAC adresu.


Wykonuję programy na zamówienie, pisać na Priv.
Asm/C/C++/Pascal/Delphi/Java/C#/PHP/JS oraz inne języki.
freemp3
  • Rejestracja:ponad 11 lat
  • Ostatnio:około 15 godzin
  • Lokalizacja:Miechów
  • Postów:284
0

Po pierwsze do czego dokładnie chcesz to wykorzystać?
Po drugie o jakiej sieci mowa? Lokalnej, firmowej, globalnej?


flowCRANE
Moderator Delphi/Pascal
  • Rejestracja:ponad 13 lat
  • Ostatnio:około 4 godziny
  • Lokalizacja:Tuchów
  • Postów:12171
0

@Kaska1988 - jeśli potrzebujesz mieć unikalny numer, to skorzystaj z unikalnych elementów; Ani adres IP, ani nazwa użytkownika nimi nie są, nawet MAC adres karty sieciowej może się powtarzać (można go komuś zwędzić), choć to bardzo rzadka sytuacja; Poza tym jeden komputer może mieć wiele kart sieciowych, stąd wiele adresów MAC - trzeba by ustalić jak to wykonać; Co nie zmienia faktu, że rozróżnianie po MACu to jeden z lepszych pomysłów;

Możesz też sprawdzać numery seryjne podzespołów - dysku twardego, płyty głównej, one raczej się nie powtarzają; Albo najlepiej łączyć kilka z nich i tworzyć jakiś hash (miksturę z różnych danych), będzie mniejsza szansa zdublowania numeru wynikowego.


Pracuję nad własną, arcade'ową, docelowo komercyjną grą z gatunku action/adventure w stylu retro (pixel art), programując silnik i powłokę gry od zupełnych podstaw, przy użyciu Free Pascala i SDL3. Więcej informacji znajdziesz na moim mikroblogu.
edytowany 2x, ostatnio: flowCRANE
TA
  • Rejestracja:ponad 11 lat
  • Ostatnio:około 6 lat
0

Sam MAC karty jest bardzo słabym pomysłem, bo można go dowolnie zmieniać i powstaje problem z wybraniem MACa jeśli w systemie jest zainstalowanych kilka kart (nie koniecznie fizycznych, mogą być też wirtualne). Tak jak napisał @furious programming lepiej użyć identyfikatorów sprzętu: typ/model/numer seryjny procesora, numer seryjny płyty głównej, numer seryjny dysku na którym aplikacja jest zainstalowana, numer seryjny/producent karty graficznej lub identyfikatorów systemu np. numer seryjny, numer aktywacyjny, GUID systemu, wersja, etc. Mając te informacje można stworzyć HWID wykorzystując jedną z funkcji hashujących np. md5, sha. Korzystając z tej metody trzeba pamiętać żeby przynajmniej jeden z wybranych identyfikatorów był unikalny.

HWID := funkcja hashująca(jak największy zbiór identyfikatorów);

Mały przykład funkcji odczytujących identyfikatory z rejestru:

Kopiuj
function getBiosInfo: string;
begin
    Result := '';
    try
        Registry.OpenKeyReadOnly('\HARDWARE\DESCRIPTION\System\BIOS\');
        Result := Registry.ReadString('BIOSVendor');
        Result := Result + ' ' + Registry.ReadString('BIOSVersion');
    except
    end;
end;

function getCPUID: string;
begin
    Result := '';
    try
        Registry.OpenKeyReadOnly('\HARDWARE\DESCRIPTION\System\CentralProcessor\0');
        Result := Registry.ReadString('Identifier');
        Result := Result + ' ' + Registry.ReadString('ProcessorNameString');
    except
    end;
end;

function getUniqueID: string;
begin
    Result := '';
    try
        Registry.OpenKeyReadOnly('\SOFTWARE\Microsoft\Cryptography');
        Result := Registry.ReadString('MachineGuid');
    except
    end;
end;
flowCRANE
Zamykaj klucze metodą CloseKey po odczytaniu z nich informacji;
TA
Wydaje mi się, że przy samym odczycie CloseKey nie jest konieczne.
flowCRANE
Trzeba zamykać, bo zarówno OpenKey, jak i OpenKeyReadOnly zmieniają numer i ścieżkę otwartego klucza; Jeśli klucza nie zamkniesz, będzie on cały czas otwarty dla kolejnych operacji, w których podane ścieżki bezwzględne będą błędne z punktu widzenia struktury drzewa rejestru;
TA
Nie wiem jak w starszych wersjach delphi, ale w nowszych metody OpenKey i OpenKeyReadOnly automatycznie zamykają poprzednio otwarty klucz. Jeśli chcemy odczytać kilka wartości z rejestru i zaraz po odczytaniu zwolnić obiekt TRegistry to CloseKey można bezpiecznie pominąć.
flowCRANE
Jeśli tak, to Ok - zwracam honor;
Azarien
  • Rejestracja:ponad 21 lat
  • Ostatnio:około 6 godzin
1

Sam MAC karty jest bardzo słabym pomysłem, bo można go dowolnie zmieniać

To wszystko zależy w jakim celu ma być identyfikacja komputerów przeprowadzana.

Czy to jest DRM, gdzie zakładamy że każdy użytkownik to haker zdeterminowany by położyć system?
Czy może komputery są firmowe, i chcemy prowadzić ich jakąś ewidencję?

Bez odpowiedzi na pytanie "po co to" nie da się udzielić sensownej rady.

TA
Niby tak, ale identyfikator oparty na MAC nie daje gwarancji unikalności, przez co wykorzystanie go do prostej ewidencji jest dyskusyjne, że o DRM nie wspomnę.
flowCRANE
Ogólnie rzecz biorąc wszystkie podane przez pytacza informacje, które chce wykorzystać do budowy numeru są słabym pomysłem, bo nie są prawdziwie i zawsze unikalne;
0
Kopiuj
Registry.OpenKeyReadOnly('\HARDWARE\DESCRIPTION\System\BIOS\');

dupa, bo w win64 nie ma takiego klucza.

dodanie znacznika <code class="delphi"> - furious programming

edytowany 1x, ostatnio: flowCRANE
Młody
No chyba, że u Ciebie nie ma, bo normalnie jest....
flowCRANE
@dusicielak - wstawiaj kod w znaczniki kolorujące składnię;
KA
  • Rejestracja:prawie 20 lat
  • Ostatnio:około godziny
  • Lokalizacja:Gorlice
1

Jak to nie ma:

Kopiuj
uses Registry;

function getBiosInfo: string;
var
  Reg: TRegistry;
begin
  Result := '';
  Reg:= TRegistry.Create;
  try
    Reg.RootKey:= HKEY_LOCAL_MACHINE;
    Reg.OpenKeyReadOnly('\HARDWARE\DESCRIPTION\System\BIOS\');
    Result := Reg.ReadString('BIOSVendor');
    Result := Result + ' ' + Reg.ReadString('BIOSVersion');
  finally
  Reg.Free;
  end;
end;

procedure TForm1.Button1Click(Sender: TObject);
begin
  ShowMessage(getBiosInfo);
end;

Nie odpowiadam na PW w sprawie pomocy programistycznej.
Pytania zadawaj na forum, bo:
od tego ono jest ;) | celowo nie zawracasz gitary | przeczyta to więcej osób a więc większe szanse że ktoś pomoże.
Młody
U niego, nie ma, bo pewnie nie wie, że jest jeszcze coś takiego jak RootKey :)
KA
No może bo domyślnie jest ustawiony na HKEY_CURRENT_USER
Kaska1988
  • Rejestracja:prawie 12 lat
  • Ostatnio:5 miesięcy
  • Postów:186
0

Wszystko fajnie, ale:
funkcja getBiosInfo - ta funkcja działa mi tylko na Win7, pod XP nie zwraca nic
funkcje getCPUID i getUniqueId nie zwracają mi nic zarówno pod win7 (x86) i WinXP (x86) ;/


0
Młody
ciekawe, mam win7 x64 i jakoś wszystkie te klucze istnieją. Chyba, że chodzi o inną Winde.
flowCRANE
U mnie na WinXP także nie ma klucza HKEY_LOCAL_MACHINE\HARDWARE\DESCRIPTION\System\BIOS\
Młody
No, to teraz wiemy, że chodzi o WinXP.
TA
  • Rejestracja:ponad 11 lat
  • Ostatnio:około 6 lat
1

Po pierwsze: tak jak poprzednicy zauważyli, przydałaby się informacja do czego autor(ka) tematu zamierza ten HWID wykorzystać.
Po drugie: to co podałem to był tylko mały PRZYKŁAD, a nie gotowiec :P Ale skoro już zaczęliście sprawdzać dostępność to:

WinXP x86 - nie ma klucza BIOS, reszta powinna być

Win Vista x86 - wszystko powinno być
Win Vista x64 - wszystko powinno być

Win7 x86 - wszystko powinno być
Win7 x64 - wszystko powinno być

Win8 x86 - wszystko powinno być
Win8 x64 - wszystko powinno być

Systemy były co najmniej w wersji Professional ze wszystkimi dostępnymi aktualizacjami. Najważniejszy z tych kluczy - "MachineGuid" był dostępny w każdej testowanej wersji systemu, reszta może zależeć od konfiguracji sprzętowej.

Jeśli chodzi o samo generowanie HWID to nie ma znaczenia, czy wszystkie identyfikatory zostaną odczytane. Wystarczy, że mamy minimum jeden, który jest unikalny i zawsze dostępny. Finalnie i tak wszystkie powinny być hashowane.

edytowany 1x, ostatnio: Tajiri
Młody
  • Rejestracja:około 22 lata
  • Ostatnio:ponad 10 lat
  • Postów:418
0
edytowany 1x, ostatnio: Młody
0

Ja czytałem to w XP, i zawsze tam to było.

W 7 64 nie było, ale teraz patrzę i widzę że się cudownie pojawiły.
Pewnie to jakaś usługa tam zapisuje, i przed jej odpaleniem tego nie ma.

Też takiego sprzętowego numera potrzebowałem... do sprawdzania legalności softu.

Myślałem też o czytaniu wprost z dysku daty utworzenia katalogu systemowego, zwykle Windows,
Ale to jest trochę do d**y, bo po reinstalacji windows użytkownik musiałby aktywować program od nowa,
a ja nawet nie wiedziałbym czy on tego na komputerze kolegi nie aktywuje. :)

Kaska1988
  • Rejestracja:prawie 12 lat
  • Ostatnio:5 miesięcy
  • Postów:186
0

Po prostu zaczęłam pisać swój komunikator i zastanawiałam się jak rozwiązać sprawę rozróżniania komputerów. Póki co rozwiązałam to kombinacją NazwaUsera_NazwaKomputera.


TA
Napisz jeszcze w jakim celu chcesz rozróżniać te komputery. Zakładając, że ten komunikator posiada jakieś konta użytkowników czemu nie chcesz/nie możesz wykorzystać tej informacji? Planujesz zbierać dane statystyczne, śledzić użytkowników, kontrolować licencje czy jeszcze coś innego?
babubabu
Mam dziwne wrażenie, że ten komunikator będzie bez serwera zarządzającego a co za tym idzie trzeba jakoś rozróżnić kto jest "zalogowany" i stąd ten unikalny identyfikator potrzebny.
RD
  • Rejestracja:około 12 lat
  • Ostatnio:11 miesięcy
  • Postów:212
0

Wycinam jedną cześć ze swojej biblioteki. Moim zdaniem szansa na powtórkę mała.

Kopiuj
{ Pobieranie numeru seryjnego płyty głównej }
function GetMotherBoardSerial: String;
var
  objWMIService: OLEVariant;
  colItems: OLEVariant;
  colItem: OLEVariant;
  oEnum: IEnumvariant;
  iValue: LongWord;

  function GetWMIObject(const objectName: String): IDispatch;
  var
    chEaten: Integer;
    BindCtx: IBindCtx;
    Moniker: IMoniker;
  begin
    OleCheck(CreateBindCtx(0, BindCtx));
    OleCheck(MkParseDisplayName(BindCtx, StringToOleStr(objectName), chEaten, Moniker));
    OleCheck(Moniker.BindToObject(BindCtx, nil, IDispatch, Result));
  end;

begin
  Result := '';
  objWMIService := GetWMIObject('winmgmts:\\localhost\root\cimv2');
  colItems := objWMIService.ExecQuery('SELECT SerialNumber FROM Win32_BaseBoard', 'WQL', 0);
  oEnum := IUnknown(colItems._NewEnum) as IEnumvariant;
  if oEnum.Next(1, colItem, iValue) = 0 then
    Result := VarToStr(colItem.SerialNumber);
end;
edytowany 2x, ostatnio: Rafał D

Zarejestruj się i dołącz do największej społeczności programistów w Polsce.

Otrzymaj wsparcie, dziel się wiedzą i rozwijaj swoje umiejętności z najlepszymi.