123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223 |
- /* vim: set expandtab ts=4 sw=4: */
- /*
- * You may redistribute this program and/or modify it under the terms of
- * the GNU General Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- */
- #include "benc/String.h"
- #include "crypto/random/Random.h"
- #include "util/events/EventBase.h"
- #include "util/events/PipeServer.h"
- #include "util/events/Pipe.h"
- #include "util/events/Timeout.h"
- #include "memory/Allocator.h"
- #include "util/events/Process.h"
- #include "util/log/Log.h"
- #include "util/log/FileWriterLog.h"
- #include "util/CString.h"
- #include "util/Assert.h"
- #include "wire/Message.h"
- #include "wire/Error.h"
- #include <errno.h>
- #include <fcntl.h>
- #include <unistd.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #define MESSAGE "IT WORKS!"
- #define MESSAGEB "INDEED"
- struct Context {
- struct Iface iface;
- struct Allocator* alloc;
- struct Allocator* rootAlloc;
- struct EventBase* base;
- struct Log* log;
- // Parent only
- int fd;
- // child only
- char* name;
- Identity
- };
- static void onConnectionParent(struct PipeServer* p, struct Sockaddr* addr)
- {
- struct Context* c = Identity_check((struct Context*) p->userData);
- struct Allocator* alloc = Allocator_child(c->alloc);
- struct Message* msg = Message_new(0, 256, alloc);
- Er_assert(Message_epush(msg, MESSAGE, CString_strlen(MESSAGE) + 1));
- Er_assert(AddrIface_pushAddr(msg, addr));
- if (!Defined(win32)) {
- Message_setAssociatedFd(msg, c->fd);
- }
- printf("Parent sending message [%s] len [%d]\n", MESSAGE, Message_getLength(msg));
- Iface_send(&c->iface, msg);
- Allocator_free(alloc);
- }
- static int g_childStopped = 0;
- static struct Context* g_context = NULL;
- static Iface_DEFUN receiveMessageParent(struct Message* msg, struct Iface* iface)
- {
- struct Context* c = Identity_check((struct Context*) iface);
- Er_assert(AddrIface_popAddr(msg));
- Assert_true(Message_getLength(msg) == (int)CString_strlen(MESSAGEB)+1);
- Assert_true(!Bits_memcmp(msg->msgbytes, MESSAGEB, CString_strlen(MESSAGEB)+1));
- g_context = c;
- if (g_childStopped) {
- printf("Parent stopped in receiveMessageParent\n");
- Allocator_free(c->rootAlloc);
- }
- return NULL;
- }
- static void timeout(void* vNULL)
- {
- Assert_true(!"timed out.");
- }
- static void onConnectionChild(struct Pipe* p, int status)
- {
- Assert_true(!status);
- printf("Child connected\n");
- }
- static Iface_DEFUN receiveMessageChild(struct Message* msg, struct Iface* iface)
- {
- struct Context* c = Identity_check((struct Context*) iface);
- struct Message* m = Message_clone(msg, c->alloc);
- printf("Child received message\n");
- Assert_true(Message_getLength(m) == (int)CString_strlen(MESSAGE)+1);
- Assert_true(!Bits_memcmp(m->msgbytes, MESSAGE, CString_strlen(MESSAGE)+1));
- if (!Defined(win32)) {
- int fd = Message_getAssociatedFd(msg);
- if (lseek(fd, 0, SEEK_SET) < 0) {
- printf("lseek(%d) failed: errno %s\n", fd, strerror(errno));
- Assert_failure("lseek()");
- }
- uint8_t* buf = Allocator_calloc(Message_getAlloc(msg), 2048, 1);
- if (read(fd, buf, 1024) < 0) {
- printf("read(%d) failed: errno %s\n", fd, strerror(errno));
- Assert_failure("read()");
- }
- if (CString_strncmp(buf, c->name, 1024)) {
- printf("want: %s\n"
- "got: %s", c->name, buf);
- Assert_failure("file content is wrong");
- }
- }
- Er_assert(Message_eshift(m, -((int)CString_strlen(MESSAGE))));
- Er_assert(Message_epush(m, MESSAGEB, CString_strlen(MESSAGEB)));
- Iface_send(&c->iface, m);
- exit(0);
- // shutdown
- Allocator_free(c->rootAlloc);
- return NULL;
- }
- static void child(char* name, struct Context* ctx)
- {
- struct Pipe* pipe = Er_assert(Pipe_named(name, ctx->base, ctx->log, ctx->alloc));
- pipe->onConnection = onConnectionChild;
- pipe->userData = ctx;
- ctx->iface.send = receiveMessageChild;
- ctx->name = name;
- Iface_plumb(&ctx->iface, &pipe->iface);
- Timeout_setTimeout(timeout, NULL, 2000, ctx->base, ctx->alloc);
- EventBase_beginLoop(ctx->base);
- }
- static void onChildExit(int64_t exit_status, int term_signal)
- {
- printf("\n\n\nChild process exit status = %d term_signal = %d\n\n\n",
- (int)exit_status, term_signal);
- Assert_true(exit_status == 0);
- g_childStopped = true;
- if (g_context) {
- printf("Parent stopped in onChildExit\n");
- Allocator_free(g_context->rootAlloc);
- }
- }
- int main(int argc, char** argv)
- {
- struct Allocator* allocator = Allocator_new(1<<20);
- struct EventBase* eb = EventBase_new(allocator);
- struct Allocator* alloc = Allocator_child(allocator);
- struct Log* log = FileWriterLog_new(stdout, alloc);
- struct Context* ctx = Allocator_calloc(alloc, sizeof(struct Context), 1);
- Identity_set(ctx);
- ctx->alloc = alloc;
- ctx->rootAlloc = allocator;
- ctx->base = eb;
- ctx->log = log;
- ctx->iface.send = receiveMessageParent;
- if (argc > 3 && !CString_strcmp("Process_test", argv[1]) && !CString_strcmp("child", argv[2])) {
- child(argv[3], ctx);
- return 0;
- }
- struct Random* rand = Random_new(alloc, log, NULL);
- char randName[32] = {0};
- Random_base32(rand, (uint8_t*)randName, 31);
- String* name = String_printf(alloc, "%s%scjdns-test-%s", Pipe_PATH, Pipe_PATH_SEP, randName);
- if (!Defined(win32)) {
- String* textName =
- String_printf(alloc, "%s%scjdns-test-%s.txt", Pipe_PATH, Pipe_PATH_SEP, randName);
- int fd = open(textName->bytes, O_CREAT | O_TRUNC | O_RDWR, 0600);
- Assert_true(fd >= 0);
- Assert_true(write(fd, name->bytes, name->len) == ((ssize_t)name->len));
- ctx->fd = fd;
- unlink(textName->bytes);
- }
- struct PipeServer* pipe = PipeServer_named(name->bytes, eb, NULL, log, alloc);
- pipe->userData = ctx;
- pipe->onConnection = onConnectionParent;
- Iface_plumb(&ctx->iface, pipe->iface.iface);
- const char* path = Process_getPath(alloc);
- Assert_true(path != NULL);
- #ifdef win32
- Assert_true(CString_strstr(path, ":\\") == path + 1); /* C:\ */
- Assert_true(CString_strstr(path, ".exe"));
- #elif openbsd
- Assert_true(path[0] == 'b'); // Process_getPath returns relative paths on openbsd
- #elif netbsd
- Assert_true(path[0] == 'b'); // Process_getPath returns relative paths on netbsd too
- #else
- Assert_true(path[0] == '/');
- #endif
- const char* args[] = { "Process_test", "child", name->bytes, NULL };
- Assert_true(!Process_spawn(path, args, eb, alloc, onChildExit));
- Timeout_setTimeout(timeout, NULL, 2000, eb, alloc);
- EventBase_beginLoop(eb);
- unlink(name->bytes);
- return 0;
- }
|