Game-Tibia-Packet
view release on metacpan or search on metacpan
lib/Game/Tibia/Packet/Login.pm view on Meta::CPAN
use strict;
use warnings;
no warnings 'uninitialized';
use v5.16.0;
package Game::Tibia::Packet::Login;
# ABSTRACT: Login packet support for the MMORPG Tibia
our $VERSION = '0.007'; # VERSION
use Carp;
use File::ShareDir 'dist_file';
use Crypt::OpenSSL::RSA;
use Digest::Adler32;
use Game::Tibia::Packet;
use Scalar::Util qw(blessed);
use constant GET_CHARLIST => 0x01;
use constant LOGIN_CHAR => 0x0A;
=pod
=encoding utf8
=head1 NAME
Game::Tibia::Packet::Login - Login packet support for the MMORPG Tibia
=head1 SYNOPSIS
use Game::Tibia::Packet::Login;
=head1 DESCRIPTION
Decodes Tibia Login packets into hashes and vice versa. By default uses the OTServ RSA key, but allows different RSA keys to be supplied. Version 9.80 and above is not supported.
=cut
our %params;
sub import {
(undef, %params) = (shift, %params, @_);
die "Malformed Tibia version\n" if exists $params{tibia} && $params{tibia} !~ /^\d+$/;
}
my $otserv = Crypt::OpenSSL::RSA->new_private_key(
do { local $/; open my $rsa, '<', dist_file('Game-Tibia-Packet', 'otserv.private') or die "Couldn't open private key $!"; <$rsa>; }
);
=head1 METHODS AND ARGUMENTS
=over 4
=item new(version => $version, [$character => undef, packet => $packet, rsa => OTSERV])
Constructs a new C<Game::Tibia::Packet::Login> instance of version C<$version>. If C<packet> is supplied, decryption using the supplied rsa private key is attempted. If no C<rsa> is supplied, the OTServ RSA key is used. If a C<$character> name is sup...
=cut
sub new {
my $class = shift;
my $self = {
packet => undef,
rsa => $otserv,
@_
};
$self->{version} //= $self->{versions}{client}{VERSION};
$self->{version} //= $params{tibia};
croak 'A protocol version < 9.80 must be supplied' if !defined $self->{version} || $self->{version} >= 980;
$self->{versions}{client} = Game::Tibia::Packet::version($self->{version});
if ($self->{versions}{client}{rsa}) {
if (defined $self->{rsa} and !blessed $self->{rsa}) {
$self->{rsa} = Crypt::OpenSSL::RSA->new_private_key($self->{rsa});
}
$self->{rsa}->use_no_padding if defined $self->{rsa};
}
if (defined $self->{packet}) {
(my $len, my $cmd, $self->{os}, $self->{versions}{client}{VERSION}, my $payload)
= unpack 'v C (S S)< a*', $self->{packet};
croak "Expected GET_CHARLIST (0x01) or LOGIN_CHAR (0x0A) packet type, but got $cmd" if $cmd ne GET_CHARLIST and $cmd ne LOGIN_CHAR;
if ($cmd == GET_CHARLIST) {
($self->{versions}{spr}, $self->{versions}{dat}, $self->{versions}{pic}, $payload)
= unpack('(L3)< a*', $payload);
}
if ($self->{versions}{client}{rsa}) {
$payload = $self->{rsa}->decrypt($payload);
croak q(Decoded RSA doesn't start with zero.) if $payload !~ /^\0/;
$payload = substr $payload, 1;
}
if ($self->{versions}{client}{xtea}) {
($self->{xtea}, $payload) = unpack 'a16 a*', $payload;
}
if ($cmd == LOGIN_CHAR) {
($self->{gmflag}, $payload) = unpack "C a*", $payload;
}
my $acc_data_pattern = $self->{versions}{client}{acc_name} ? '(S/a)<' : 'V';
($self->{account}, $payload) = unpack "$acc_data_pattern a*", $payload;
if ($cmd == LOGIN_CHAR) {
($self->{character}, $payload) = unpack "(S/a)< a*", $payload;
}
($self->{password}, $payload) = unpack "(S/a)< a*", $payload;
if ($cmd == LOGIN_CHAR) {
($self->{nonce}, $payload) = unpack "(a5) a*", $payload;
}
$self->{undecoded} = unpack "a*", $payload;
}
bless $self, $class;
return $self;
}
=item finalize([$rsa])
Finalizes the packet. encrypts with RSA and prepends header
=cut
sub finalize {
my $self = shift;
my $rsa = shift // $self->{rsa};
$self->{rsa}->use_no_padding if defined $self->{rsa};
$self->{versions}{client} = Game::Tibia::Packet::version $self->{versions}{client} unless ref $self->{versions}{client};
my $payload = '';
if ($self->{versions}{client}{rsa}) {
$rsa = Crypt::OpenSSL::RSA->new_private_key($rsa) unless blessed $rsa;
$rsa->size == 128
or croak "Protocol $self->{versions}{client}{VERSION} expects 128 bit RSA key, but ${\($rsa->size*8)} bit were provided";
$payload .= "\0";
}
$self->{packet} = defined $self->{character} ? "\x0a" : "\x01";
$self->{packet} .= pack '(S2)<', $self->{os}, $self->{versions}{client}{VERSION};
$self->{packet} .= defined $self->{character} ? "\0" :
pack '(L3)<', $self->{versions}{spr}, $self->{versions}{dat}, $self->{versions}{pic};
my $acc_pattern = $self->{versions}{client}{acc_name} ? '(S/a)<' : 'V';
$payload .= $self->{xtea} if $self->{versions}{client}{xtea};
$payload .= pack "C", $self->{gmflag} if defined $self->{gmflag};
$payload .= pack $acc_pattern, $self->{account};
$payload .= pack '(S/a)<', $self->{character} if defined $self->{character};
$payload .= pack '(S/a)<', $self->{password};
$payload .= pack 'a5', $self->{nonce} if defined $self->{nonce};
$payload .= pack 'a*', $self->{undecoded} if defined $self->{undecoded} && $self->{undecoded} ne '';
if ($self->{versions}{client}{rsa}) {
my $padding_len = 128 - length($payload);
$payload .= pack "a$padding_len", '';
$payload = $self->{rsa}->encrypt($payload);
}
$self->{packet} .= $payload;
if ($self->{versions}{client}{adler32}) {
my $a32 = Digest::Adler32->new;
$a32->add($self->{packet});
my $digest = pack "N", unpack "L", $a32->digest;
$self->{packet} = $digest.$self->{packet};
}
$self->{packet} = pack("(S/a)<", $self->{packet});
$self->{packet};
}
1;
__END__
=back
=head1 GIT REPOSITORY
L<http://github.com/athreef/Game-Tibia-Packet>
=head1 SEE ALSO
L<Game::Tibia::Packet>
L<Game::Tibia::Packet::Charlist>
=head1 AUTHOR
Ahmad Fatoum C<< <athreef@cpan.org> >>, L<http://a3f.at>
( run in 0.890 second using v1.01-cache-2.11-cpan-13bb782fe5a )