/* * Copyright (C) 2004-2023 ZNC, see the NOTICE file for details. * * 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. */ #include "znctest.h" namespace znc_inttest { namespace { TEST_F(ZNCTest, Modperl) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modperl"); client.Write("znc loadmod perleval"); client.Write("PRIVMSG *perleval :2+2"); client.ReadUntil(":*perleval!znc@znc.in PRIVMSG nick :Result: 4"); client.Write("PRIVMSG *perleval :$self->GetUser->GetUsername"); client.ReadUntil("Result: user"); } TEST_F(ZNCTest, Modpython) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modpython"); client.Write("znc loadmod pyeval"); client.Write("PRIVMSG *pyeval :2+2"); client.ReadUntil(":*pyeval!znc@znc.in PRIVMSG nick :4"); client.Write("PRIVMSG *pyeval :module.GetUser().GetUsername()"); client.ReadUntil("nick :'user'"); ircd.Write(":server 001 nick :Hello"); ircd.Write(":n!u@h PRIVMSG nick :Hi\xF0, github issue #1229"); // "replacement character" client.ReadUntil("Hi\xEF\xBF\xBD, github issue"); // Non-existing encoding client.Write("PRIVMSG *controlpanel :Set ClientEncoding $me Western"); client.Write("JOIN #a\342"); client.ReadUntil( ":*controlpanel!znc@znc.in PRIVMSG nick :ClientEncoding = UTF-8"); ircd.ReadUntil("JOIN #a\xEF\xBF\xBD"); } TEST_F(ZNCTest, ModpythonSocket) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("socktest.py", R"( import znc class acc(znc.Socket): def OnReadData(self, data): self.GetModule().PutModule('received {} bytes'.format(len(data))) self.Close() class lis(znc.Socket): def OnAccepted(self, host, port): sock = self.GetModule().CreateSocket(acc) sock.DisableReadLine() return sock class socktest(znc.Module): def OnLoad(self, args, ret): listen = self.CreateSocket(lis) self.port = listen.Listen() return True def OnModCommand(self, cmd): sock = self.CreateSocket() sock.Connect('127.0.0.1', self.port) sock.WriteBytes(b'blah') )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modpython"); client.Write("znc loadmod socktest"); client.Write("PRIVMSG *socktest :foo"); client.ReadUntil("received 4 bytes"); } TEST_F(ZNCTest, ModperlSocket) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("socktest.pm", R"( package socktest::acc; use base 'ZNC::Socket'; sub OnReadData { my ($self, $data, $len) = @_; $self->GetModule->PutModule("received $len bytes"); $self->Close; } package socktest::lis; use base 'ZNC::Socket'; sub OnAccepted { my $self = shift; return $self->GetModule->CreateSocket('socktest::acc'); } package socktest::conn; use base 'ZNC::Socket'; package socktest; use base 'ZNC::Module'; sub OnLoad { my $self = shift; my $listen = $self->CreateSocket('socktest::lis'); $self->{port} = $listen->Listen; return 1; } sub OnModCommand { my ($self, $cmd) = @_; my $sock = $self->CreateSocket('socktest::conn'); $sock->Connect('127.0.0.1', $self->{port}); $sock->Write('blah'); } 1; )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modperl"); client.Write("znc loadmod socktest"); client.Write("PRIVMSG *socktest :foo"); client.ReadUntil("received 4 bytes"); } TEST_F(ZNCTest, ModpythonVCString) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("test.py", R"( import znc class test(znc.Module): def OnUserRawMessage(self, msg): self.PutModule(str(msg.GetParams())) return znc.CONTINUE )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modpython"); client.Write("znc loadmod test"); client.Write("PRIVMSG *test :foo"); client.ReadUntil("'*test', 'foo'"); } TEST_F(ZNCTest, ModperlVCString) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("test.pm", R"( package test; use base 'ZNC::Module'; sub OnUserRawMessage { my ($self, $msg) = @_; my @params = $msg->GetParams; $self->PutModule("@params"); return $ZNC::CONTINUE; } 1; )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modperl"); client.Write("znc loadmod test"); client.Write("PRIVMSG *test :foo"); client.ReadUntil(":*test foo"); } TEST_F(ZNCTest, ModperlNV) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("test.pm", R"( package test; use base 'ZNC::Module'; sub OnLoad { my $self = shift; $self->SetNV('a', 'X'); $self->NV->{b} = 'Y'; my @k = keys %{$self->NV}; $self->PutModule("@k"); return $ZNC::CONTINUE; } 1; )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modperl"); client.Write("znc loadmod test"); client.ReadUntil(":a b"); } TEST_F(ZNCTest, ModpythonPackage) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); QDir dir(m_dir.path()); ASSERT_TRUE(dir.mkpath("modules")); ASSERT_TRUE(dir.cd("modules")); ASSERT_TRUE(dir.mkpath("packagetest")); InstallModule("packagetest/__init__.py", R"( import znc from .data import value class packagetest(znc.Module): def OnModCommand(self, cmd): self.PutModule('value = ' + value) )"); InstallModule("packagetest/data.py", "value = 'a'"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modpython"); client.Write("znc loadmod packagetest"); client.Write("PRIVMSG *packagetest :foo"); client.ReadUntil("value = a"); InstallModule("packagetest/data.py", "value = 'b'"); client.Write("PRIVMSG *packagetest :foo"); client.ReadUntil("value = a"); client.Write("znc updatemod packagetest"); client.Write("PRIVMSG *packagetest :foo"); client.ReadUntil("value = b"); } TEST_F(ZNCTest, ModpythonModperl) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); auto ircd = ConnectIRCd(); auto client = LoginClient(); // https://github.com/znc/znc/issues/1757 client.Write("znc loadmod modpython"); client.ReadUntil("Loaded module modpython"); client.Write("znc loadmod modperl"); client.ReadUntil("Loaded module modperl"); } TEST_F(ZNCTest, ModpythonCommand) { if (QProcessEnvironment::systemEnvironment().value( "DISABLED_ZNC_PERL_PYTHON_TEST") == "1") { return; } auto znc = Run(); znc->CanLeak(); InstallModule("cmdtest.py", R"( import znc class cmdtest(znc.Module): def OnLoad(self, args, message): self.AddHelpCommand() self.AddCommand(testcmd) return True class testcmd(znc.Command): command = 'ping' args = cmdtest.t_d('ar') description = cmdtest.t_d('blah') def __call__(self, line): self.GetModule().PutModule(line + cmdtest.t_s(' pong')) )"); auto ircd = ConnectIRCd(); auto client = LoginClient(); client.Write("znc loadmod modpython"); client.Write("znc loadmod cmdtest"); client.Write("PRIVMSG *cmdtest :ping or"); client.ReadUntil(":*cmdtest!znc@znc.in PRIVMSG nick :ping or pong"); InstallTranslation("cmdtest", "ru_RU", R"( msgid "" msgstr "" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=4; plural=((n%10==1 && n%100!=11) ? 0 : ((n%10 >= 2 " "&& n%10 <=4 && (n%100 < 12 || n%100 > 14)) ? 1 : ((n%10 == 0 || (n%10 >= 5 " "&& n%10 <=9)) || (n%100 >= 11 && n%100 <= 14)) ? 2 : 3));\n" "Language: ru_RU\n" msgid "ar" msgstr "аргумент" msgid "blah" msgstr "бла" msgid " pong" msgstr " понг" )"); client.Write("PRIVMSG *controlpanel :set language $me ru-RU"); client.Write("PRIVMSG *cmdtest :help"); client.ReadUntil(":*cmdtest!znc@znc.in PRIVMSG nick :\x02ping аргумент\x0F: бла"); client.Write("PRIVMSG *cmdtest :ping"); client.ReadUntil(":*cmdtest!znc@znc.in PRIVMSG nick :ping понг"); } } // namespace } // namespace znc_inttest