shithub: mc

ref: 0226a1a631e50ac5ad042efc85cf5d5461f5b0f8
dir: /libstd/dial.myr/

View raw version
use "alloc.use"
use "die.use"
use "error.use"
use "sys.use"
use "sleq.use"
use "option.use"
use "ipparse.use"
use "resolve.use"
use "fmt.use"
use "endian.use"
use "intparse.use"
use "hasprefix.use"
use "utf.use"

pkg std =
	const dial	: (dialstr : byte[:] -> error(fd, byte[:]))
;;

/*
 a map from service name to a list of (port,proto)
 pairs in order of preference
*/
/* FIXME: implement
var services : htab(byte[:], [int, byte[:]][:])#
var inited = false
*/

/* takes a plan 9 style dial string */
const dial = {str
	var proto, host, port
	var socktype, portnum
	var sa : sockaddr_in	/* we only support inet sockets right now.. ugh. */
	var sock

	(proto, str) = nameseg(str)
	(host, str) = nameseg(str)
	(port, str) = nameseg(str)

	if proto.len == 0
		-> `Failure "missing proto"
	elif host.len == 0
		-> `Failure "missing host"
	elif port.len == 0
		-> `Failure "missing port"
	;;

	if sleq(proto, "net")
		-> `Failure "net wildcard proto not yet supported\n"
	elif sleq(proto, "unix")
		-> `Failure "net unix proto not yet supported\n"
	elif sleq(proto, "tcp")
		socktype = Sockstream
	elif sleq(proto, "udp")
		socktype = Sockdgram
	;;

	match parseport(port)
	| `Some n:	portnum = n
	| `None:	-> `Failure "bad port"
	;;

	match getaddr(host)
	| `Ipv4 bits:
		put("connecting to %ub.%ub.%ub.%ub:%i\n", bits[0], bits[1], bits[2], bits[3], portnum)
		sa.fam = Afinet
		sa.addr = bits
		sa.port = hosttonet(portnum)
	| `Ipv6 bits:
		-> `Failure "ipv6 not yet supported"
	;;

	put("socketing...\n")
	sock = socket(sa.fam, socktype, 0)
	put("socketed\n")
	if sock < 0
		-> `Failure "failed to connect to socket"
	;;
	var err
	put("connecting...\n")
	err = connect(sock, (&sa) castto(sockaddr#), sizeof(sockaddr_in))
	put("connected\n")
	if err < 0
		put("Errno %i\n", -err)
		close(sock)
		-> `Failure "Failed to bind socket"
	;;

	-> `Success sock
}

const parseport = {port
	match intparse(port)
	| `Some n:	-> `Some n
	| `None:
		/* a small number of hardcoded ports */
		if sleq(port, "http")
			-> `Some 80
		elif sleq(port, "https")
			-> `Some 443
		elif sleq(port, "ircd")
			-> `Some 6667
		elif sleq(port, "dns")
			-> `Some 53
		;;
	;;
	-> `None
}

const getaddr = {addr
	var ip

	match ipparse(addr)
	| `Some a:	ip = a
	| `None:
		match resolve(addr)
		| `Success hi:
			ip = hi[0].addr
			slfree(hi)
		| `Failure m:
		;;
	;;
	-> ip
}

const nameseg = {str
	var len

	for len = 0; len < str.len; len++
		if str[len] == '!' castto(byte)
			-> (str[:len], str[len+1:])
		;;
	;;
	-> (str[:], str[len:])
}