299 lines
7.9 KiB
C
299 lines
7.9 KiB
C
/* $NetBSD: api_bsd.c,v 1.7 1998/11/06 20:00:07 christos Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 1988 The Regents of the University of California.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by the University of
|
|
* California, Berkeley and its contributors.
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
#ifndef lint
|
|
#if 0
|
|
static char sccsid[] = "@(#)api_bsd.c 4.2 (Berkeley) 4/26/91";
|
|
#else
|
|
__RCSID("$NetBSD: api_bsd.c,v 1.7 1998/11/06 20:00:07 christos Exp $");
|
|
#endif
|
|
#endif /* not lint */
|
|
|
|
#if defined(unix)
|
|
|
|
#ifdef __STDC__
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#else
|
|
extern char *getenv();
|
|
extern char *gepass();
|
|
#endif
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/in.h>
|
|
#include <netdb.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "../ctlr/api.h"
|
|
#include "api_exch.h"
|
|
|
|
|
|
int
|
|
api_close_api()
|
|
{
|
|
if (api_exch_outcommand(EXCH_CMD_DISASSOCIATE) == -1) {
|
|
return -1;
|
|
} else if (api_exch_flush() == -1) {
|
|
return -1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
int
|
|
api_open_api(string)
|
|
char *string; /* if non-zero, where to connect to */
|
|
{
|
|
struct sockaddr_in server;
|
|
struct hostent *hp;
|
|
struct storage_descriptor sd;
|
|
char thehostname[100];
|
|
char keyname[100];
|
|
char inkey[100];
|
|
FILE *keyfile;
|
|
int sock;
|
|
unsigned int port;
|
|
int i;
|
|
|
|
if (string == 0) {
|
|
string = getenv("API3270"); /* Get API */
|
|
if (string == 0) {
|
|
fprintf(stderr,
|
|
"API3270 environmental variable not set - no API.\n");
|
|
return -1; /* Nothing */
|
|
}
|
|
}
|
|
|
|
if (sscanf(string, "%[^:]:%d:%s", thehostname,
|
|
(int *)&port, keyname) != 3) {
|
|
fprintf(stderr, "API3270 environmental variable has bad format.\n");
|
|
return -1;
|
|
}
|
|
/* Now, try to connect */
|
|
sock = socket(AF_INET, SOCK_STREAM, 0);
|
|
if (sock < 0) {
|
|
perror("opening API socket");
|
|
return -1;
|
|
}
|
|
server.sin_family = AF_INET;
|
|
hp = gethostbyname(thehostname);
|
|
if (hp == 0) {
|
|
fprintf(stderr, "%s specifies bad host name.\n", string);
|
|
return -1;
|
|
}
|
|
if (sizeof(server.sin_addr.s_addr) < hp->h_length)
|
|
hp->h_length = sizeof(server.sin_addr.s_addr);
|
|
(void)memcpy(&server.sin_addr.s_addr, hp->h_addr, hp->h_length);
|
|
server.sin_port = htons(port);
|
|
|
|
if (connect(sock, (struct sockaddr *)&server, sizeof server) < 0) {
|
|
perror("connecting to API server");
|
|
return -1;
|
|
}
|
|
/* Now, try application level connection */
|
|
if (api_exch_init(sock, "client") == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outcommand(EXCH_CMD_ASSOCIATE) == -1) {
|
|
return -1;
|
|
}
|
|
keyfile = fopen(keyname, "r");
|
|
if (keyfile == 0) {
|
|
perror("fopen");
|
|
return -1;
|
|
}
|
|
if (fscanf(keyfile, "%s\n", inkey) != 1) {
|
|
perror("fscanf");
|
|
return -1;
|
|
}
|
|
sd.length = strlen(inkey)+1;
|
|
if (api_exch_outtype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_BYTES, sd.length, inkey) == -1) {
|
|
return -1;
|
|
}
|
|
while ((i = api_exch_nextcommand()) != EXCH_CMD_ASSOCIATED) {
|
|
int passwd_length;
|
|
char *passwd;
|
|
char buffer[200];
|
|
|
|
switch (i) {
|
|
case EXCH_CMD_REJECTED:
|
|
if (api_exch_intype(EXCH_TYPE_STORE_DESC,
|
|
sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_intype(EXCH_TYPE_BYTES, sd.length, buffer) == -1) {
|
|
return -1;
|
|
}
|
|
buffer[sd.length] = 0;
|
|
fprintf(stderr, "%s\n", buffer);
|
|
if (api_exch_outcommand(EXCH_CMD_ASSOCIATE) == -1) {
|
|
return -1;
|
|
}
|
|
break;
|
|
case EXCH_CMD_SEND_AUTH:
|
|
if (api_exch_intype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_intype(EXCH_TYPE_BYTES, sd.length, buffer) == -1) {
|
|
return -1;
|
|
}
|
|
buffer[sd.length] = 0;
|
|
passwd = getpass(buffer); /* Go to terminal */
|
|
passwd_length = strlen(passwd);
|
|
if (api_exch_intype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_intype(EXCH_TYPE_BYTES, sd.length, buffer) == -1) {
|
|
return -1;
|
|
}
|
|
buffer[sd.length] = 0;
|
|
if (sd.length) {
|
|
char *ptr;
|
|
|
|
ptr = passwd;
|
|
i = 0;
|
|
while (*ptr) {
|
|
*ptr++ ^= buffer[i++];
|
|
if (i >= sd.length) {
|
|
i = 0;
|
|
}
|
|
}
|
|
}
|
|
sd.length = passwd_length;
|
|
if (api_exch_outcommand(EXCH_CMD_AUTH) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_BYTES, passwd_length, passwd) == -1) {
|
|
return -1;
|
|
}
|
|
break;
|
|
case -1:
|
|
return -1;
|
|
default:
|
|
fprintf(stderr,
|
|
"Waiting for connection indicator, received 0x%x.\n", i);
|
|
break;
|
|
}
|
|
}
|
|
/* YEAH */
|
|
return 0; /* Happiness! */
|
|
}
|
|
|
|
|
|
int
|
|
api_exch_api(regs, sregs, parms, length)
|
|
union REGS *regs;
|
|
struct SREGS *sregs;
|
|
char *parms;
|
|
int length;
|
|
{
|
|
struct storage_descriptor sd;
|
|
int i;
|
|
|
|
if (api_exch_outcommand(EXCH_CMD_REQUEST) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_REGS, sizeof *regs, (char *)regs) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_SREGS, sizeof *sregs, (char *)sregs) == -1) {
|
|
return -1;
|
|
}
|
|
sd.length = length;
|
|
sd.location = (long) parms;
|
|
if (api_exch_outtype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_BYTES, length, parms) == -1) {
|
|
return -1;
|
|
}
|
|
while ((i = api_exch_nextcommand()) != EXCH_CMD_REPLY) {
|
|
switch (i) {
|
|
case EXCH_CMD_GIMME:
|
|
if (api_exch_intype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd)
|
|
== -1) {
|
|
return -1;
|
|
}
|
|
/*XXX validity check GIMME? */
|
|
if (api_exch_outcommand(EXCH_CMD_HEREIS) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd)
|
|
== -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_outtype(EXCH_TYPE_BYTES, sd.length,
|
|
(char *)sd.location) == -1) {
|
|
return -1;
|
|
}
|
|
break;
|
|
case EXCH_CMD_HEREIS:
|
|
if (api_exch_intype(EXCH_TYPE_STORE_DESC, sizeof sd, (char *)&sd)
|
|
== -1) {
|
|
return -1;
|
|
}
|
|
/* XXX Validty check HEREIS? */
|
|
if (api_exch_intype(EXCH_TYPE_BYTES, sd.length,
|
|
(char *)sd.location) == -1) {
|
|
return -1;
|
|
}
|
|
break;
|
|
default:
|
|
fprintf(stderr, "Waiting for reply command, we got command %d.\n",
|
|
i);
|
|
return -1;
|
|
}
|
|
}
|
|
if (api_exch_intype(EXCH_TYPE_REGS, sizeof *regs, (char *)regs) == -1) {
|
|
return -1;
|
|
}
|
|
if (api_exch_intype(EXCH_TYPE_SREGS, sizeof *sregs, (char *)sregs) == -1) {
|
|
return -1;
|
|
}
|
|
/* YEAH */
|
|
return 0; /* Happiness! */
|
|
}
|
|
|
|
#endif /* unix */
|