When you contacts.resolveUsername, the result always contains both id and access_hash. It creates confusion for API users, as to why you are only able to resolve a user/chat/channel only by their username or by their (id, access_hash) pair (eg. here and here).
I tried searching the telegram API homepage for info, but found no info about the meaning of access_hash.
I really am confused as to why you would need something more than id (or username). I'd like to know the nature of this access_hash:
contacts.resolveUsername from account #1 and then from #2, do I
get the same number?)?I know libs like pyrogram, telethon store this access_hash inside their local sqlite databases. This way they make it possible to call high level functions which require both id and access_hash using only id.
The access_hash values commonly appear in Telegram's API when access to something should be "restricted" in some way. You can find access_hash for users, channels, or even media objects.
I'll be using Telethon for the examples below, but the same applies to any library interacting with Telegram's API.
Does it change over time? Probably not. To the best of my knowledge, the access_hash does not change over time. I have not heard of any reports from users claiming that the access_hash has changed.
It is impossible to know for sure, as that would require access to the code the Telegram servers are using for this (and they could change their implementation at any time), but I've been working with Telegram for a very long time, and I can confidently say it likely never changes.
2025-02-08 update: At some point, the official documentation for Peer database - Access hash added these paragraphs (emphasis mine):
Access hashes may not be reused across different accounts or different sessions of the same account. This is a core spam prevention feature of Telegram.
Clients and client APIs should avoid exposing access hashes to users, as they cannot be reused outside of the current session, and the user should not be burdened with storing them, when the client can perfectly do the job by itself.
So it seems reasonable to assume it doesn't change within a session, but logging in again would lead to entirely new hashes.
Does it mean anything at all? Not by itself. It's just a random-looking number. However, it is proof that you have access to a particular object, whether that's an user, channel, or media.
If the access_hash didn't exist, you could try guessing random IDs, which would for example make it possible to enumerate every user registered to Telegram!:
for user_id in range(1_000_000):
    await client.get_entity(user_id)
Thankfully, the above code won't work, because in order to fetch user information, the access_hash needs to be known. But since it's random, and different for each account, it's pretty much impossible to guess. Thus, the access_hash keeps your account safe (as long as there's no way to "reach" it by other means, e.g. via a message forwarded from you, or your participation in public groups).
Is it safe to store in the database and be certain that it's constant? Yes. Telethon v1 for example stores the access_hash inside its .session file, which lets you use just the id to refer to users and channels, as long as the library has "seen" (and cached) their access_hash before.
If you've saved an access_hash before, you can reuse it later on your own:
from telethon import types
CHANNEL = types.InputPeerChannel(channel_id=..., access_hash=...)
Does it differ from account to account? Yes. The access_hash is unique to each account. For example, if Account_A fetches Channel_X, it may have id=123, access_hash=456. But if Account_B fetches the same Channel_X, it may have id=123, access_hash=789.
This means you cannot fetch an access_hash in Account_A and then try using it in Account_B, as it won't work (stickers seem to have been an exception at some point).
Every account will see the same ID for the same thing (messages appear to be an exception, but in reality they follow the same rules; they're duplicated for each account unless they occur inside a channel, so the "same" message can appear to have a different ID.)
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With