I agree that #1 is dangerous, and I currently take advantage of the fact that all fdos are assumed to be legit. #3 is extremely inelegant, I don't like restricting the user like that.
I think we should consider the API of terminal? separate from #4.
As discussed recently WRI to another example like this, we shouldn't make the user dance to feed a plain fd to terminal?, especially to go all the way to making a port. Especially since fd->*port calls dup() and returns that new fd in that port, the user would then be responsible for the gratuitously created port and its new fd.
I don't see it being really troublesome to document that terminal? can take all three arguments, and the code is quite easy, since in all cases the underlying %isatty takes an integer fd, I just add back a clause in a cond.
#4 minus changing terminal to only take ports could be the slickest, although, again, it wouldn't be reading from fd 7, but a dup()ed fd of 7, unless we change that behavior when handed an integer instead of an fdo to not do the dup(), which is again simple in the code.
- Harold