All Downloads are FREE. Search and download functionalities are using the official Maven repository.

jnr.posix.JavaPOSIX Maven / Gradle / Ivy

There is a newer version: 4.15.102
Show newest version
package jnr.posix;

import jnr.constants.platform.Errno;
import jnr.constants.platform.Fcntl;
import jnr.constants.platform.Sysconf;
import jnr.ffi.Pointer;
import jnr.posix.util.Java5ProcessMaker;
import jnr.posix.util.MethodName;
import jnr.posix.util.Platform;
import jnr.posix.util.ProcessMaker;
import sun.misc.Signal;

import java.io.*;
import java.nio.ByteBuffer;
import java.nio.charset.Charset;
import java.util.Collection;
import java.util.Map;

import static jnr.constants.platform.Errno.EINVAL;
import static jnr.constants.platform.Errno.ENOENT;

final class JavaPOSIX implements POSIX {
    private final POSIXHandler handler;
    private final JavaLibCHelper helper;

    JavaPOSIX(POSIXHandler handler) {
        this.handler = handler;
        this.helper = new JavaLibCHelper(handler);
    }

    public ProcessMaker newProcessMaker(String... command) {
        return new Java5ProcessMaker(handler, command);
    }

    public ProcessMaker newProcessMaker() {
        return new Java5ProcessMaker(handler);
    }
    
    public FileStat allocateStat() {
        return new JavaFileStat(this, handler);
    }

    public MsgHdr allocateMsgHdr() {
        handler.unimplementedError(MethodName.getCallerMethodName());
        return null;
    }

    public SocketMacros socketMacros() {
        handler.unimplementedError(MethodName.getCallerMethodName());
        return null;
    }

    public int chmod(String filename, int mode) {
        return helper.chmod(filename, mode);
    }

    public int fchmod(int fd, int mode) {
        handler.unimplementedError("No fchmod in Java (yet)");
        return -1;
    }

    public int chown(String filename, int user, int group) {
        return helper.chown(filename, user, group);
    }

    public int fchown(int fd, int user, int group) {
        handler.unimplementedError("No fchown in Java (yet)");
        return -1;
    }

    public CharSequence crypt(CharSequence key, CharSequence salt) {
        return helper.crypt(key, salt);
    }

    public byte[] crypt(byte[] key, byte[] salt) {
        return helper.crypt(key, salt);
    }

    public int exec(String path, String... argv) {
        handler.unimplementedError("No exec in Java (yet)");
        
        return -1;
    }

    public int exec(String path, String[] argv, String[] envp) {
        handler.unimplementedError("No exec in Java (yet)");
        
        return -1;
    }
    
    public int execv(String path, String[] argv) {
        handler.unimplementedError("No execv in Java (yet)");
        
        return -1;
    }
    
    public int execve(String path, String[] argv, String[] envp) {
        handler.unimplementedError("No execve in Java (yet)");
        
        return -1;
    }
    
    public FileStat fstat(FileDescriptor descriptor) {
        handler.unimplementedError("fstat unimplemented");
        
        return null;
    }

    public FileStat fstat(int descriptor) {
        handler.unimplementedError("fstat unimplemented");
        
        return null;
    }

    public int fstat(int fd, FileStat stat) {
        handler.unimplementedError("fstat unimplemented");
        return -1;
    }

    public int fstat(FileDescriptor descriptor, FileStat stat) {
        handler.unimplementedError("fstat unimplemented");
        return -1;
    }

    public int getegid() {
        return LoginInfo.GID;
    }
    
    public int geteuid() {
        return LoginInfo.UID;
    }
    
    public int getgid() {
        return LoginInfo.GID;
    }

    public int getdtablesize() {
        handler.unimplementedError("getdtablesize unimplemented");
        return -1;
    }

    public String getlogin() {
        return helper.getlogin();
    }

    public int getpgid() {
        return unimplementedInt("getpgid");
    }

    public int getpgrp() {
        return unimplementedInt("getpgrp");
    }

    public int getpid() {
        return helper.getpid();
    }

    public int getppid() {
        return unimplementedInt("getppid");
    }

    public Passwd getpwent() {
        return helper.getpwent();
    }

    public Passwd getpwuid(int which) {
        return helper.getpwuid(which);
    }

    public Group getgrgid(int which) {
        handler.unimplementedError("getgrgid unimplemented");
        return null;
    }

    public Passwd getpwnam(String which) {
        handler.unimplementedError("getpwnam unimplemented");
        return null;
    }
    public Group getgrnam(String which) {
        handler.unimplementedError("getgrnam unimplemented");
        return null;
    }

    public Group getgrent() {
        handler.unimplementedError("getgrent unimplemented");
        return null;
    }

    public int setpwent() {
        return helper.setpwent();
    }

    public int endpwent() {
        return helper.endpwent();
    }

    public int setgrent() {
        return unimplementedInt("setgrent");
    }

    public int endgrent() {
        return unimplementedInt("endgrent");
    }

    public Pointer environ() {
        handler.unimplementedError("environ");

        return null;
    }

    // @see setenv for more on the environment methods
    public String getenv(String envName) {
        return helper.getEnv().get(envName);
    }

    public int getuid() {
        return LoginInfo.UID;
    }

    public int getrlimit(int resource, RLimit rlim) {
        return unimplementedInt("getrlimit");
    }

    public int getrlimit(int resource, Pointer rlim) {
        return unimplementedInt("getrlimit");
    }

    public RLimit getrlimit(int resource) {
        handler.unimplementedError("getrlimit");

        return null;
    }

    public int setrlimit(int resource, RLimit rlim) {
        return unimplementedInt("setrlimit");
    }

    public int setrlimit(int resource, Pointer rlim) {
        return unimplementedInt("setrlimit");
    }

    public int setrlimit(int resource, long rlimCur, long rlimMax) {
        return unimplementedInt("setrlimit");
    }

    public int fork() {
        return -1;
    }

    public boolean isatty(FileDescriptor fd) {
        return (fd == FileDescriptor.in
                || fd == FileDescriptor.out
                || fd == FileDescriptor.err);
    }

    public int isatty(int fd) {
        return (fd == 0
                || fd == 1
                || fd == 2) ? 1 : 0;
    }

    public int kill(int pid, int signal) {
        return unimplementedInt("kill");    // FIXME: Can be implemented
    }

    public int kill(long pid, int signal) {
        return unimplementedInt("kill");    // FIXME: Can be implemented
    }

    private static class SunMiscSignalHandler implements sun.misc.SignalHandler {
        final SignalHandler handler;
        public SunMiscSignalHandler(SignalHandler handler) {
            this.handler = handler;
        }
        
        public void handle(Signal signal) {
            handler.handle(signal.getNumber());
        }
    }
    
    public SignalHandler signal(jnr.constants.platform.Signal sig, final SignalHandler handler) {
        Signal s = new Signal(sig.name().substring("SIG".length()));
        
        sun.misc.SignalHandler oldHandler = Signal.handle(s, new SunMiscSignalHandler(handler));
        
        if (oldHandler instanceof SunMiscSignalHandler) {
            return ((SunMiscSignalHandler)oldHandler).handler;
        } else {
            return null;
        }
    }

    public int lchmod(String filename, int mode) {
        return unimplementedInt("lchmod");    // FIXME: Can be implemented
    }

    public int lchown(String filename, int user, int group) {
        return unimplementedInt("lchown");     // FIXME: Can be implemented
    }

    public int link(String oldpath, String newpath) {
        return helper.link(oldpath, newpath);
    }

    public FileStat lstat(String path) {
        FileStat stat = allocateStat();

        if (lstat(path, stat) < 0) handler.error(ENOENT, "lstat", path);
        
        return stat;
    }

    public int lstat(String path, FileStat stat) {
        return helper.lstat(path, stat);
    }

    public int mkdir(String path, int mode) {
        return helper.mkdir(path, mode);
    }

    public int rmdir(String path) {
        return helper.rmdir(path);
    }

    public String readlink(String path) throws IOException {
        // TODO: this should not be hardcoded to 256 bytes
        ByteBuffer buffer = ByteBuffer.allocateDirect(256);
        int result = helper.readlink(path, buffer, buffer.capacity());
        
        if (result == -1) return null;
        
        buffer.position(0);
        buffer.limit(result);
        return Charset.forName("ASCII").decode(buffer).toString();
    }

    public int readlink(CharSequence path, byte[] buf, int bufsize) {
        handler.unimplementedError("readlink");

        return -1;
    }

    public int readlink(CharSequence path, ByteBuffer buf, int bufsize) {
        handler.unimplementedError("readlink");

        return -1;
    }

    public int readlink(CharSequence path, Pointer bufPtr, int bufsize) {
        handler.unimplementedError("readlink");

        return -1;
    }

    // At this point the environment is not being used by any methods here.
    // getenv/setenv/unsetenv do behave properly via POSIX definitions, but 
    // it is only a storage facility at the moment.  In a future release, this
    // map will be hooked up to the methods which depend on env.
    public int setenv(String envName, String envValue, int overwrite) {
        Map env = helper.getEnv();
        
        if (envName.contains("=")) {
            handler.error(EINVAL, "setenv", envName);
            return -1;
        }
        
        // POSIX specified.  Existence is success if overwrite is 0.
        if (overwrite == 0 && env.containsKey(envName)) return 0;
        
        env.put(envName, envValue);
        
        return 0;
    }

    public FileStat stat(String path) {
        FileStat stat = allocateStat(); 

        if (helper.stat(path, stat) < 0) handler.error(ENOENT, "stat", path);
        
        return stat;
    }

    public int stat(String path, FileStat stat) {
        return helper.stat(path, stat);
    }

    public int symlink(String oldpath, String newpath) {
        return helper.symlink(oldpath, newpath);
    }

    public int setegid(int egid) {
        return unimplementedInt("setegid");
    }

    public int seteuid(int euid) {
        return unimplementedInt("seteuid");
    }

    public int setgid(int gid) {
        return unimplementedInt("setgid");
    }

    public int getpgid(int pid) {
        return unimplementedInt("getpgid");
    }

    public int setpgid(int pid, int pgid) {
        return unimplementedInt("setpgid");
    }

    public int setpgrp(int pid, int pgrp) {
        return unimplementedInt("setpgrp");
    }

    public int setsid() {
        return unimplementedInt("setsid");
    }

    public int setuid(int uid) {
        return unimplementedInt("setuid");
    }

    public int umask(int mask) {
        // TODO: We can possibly maintain an internal mask and try and apply it to individual
        // libc methods.  
        return 0;
    }

    public int unsetenv(String envName) {
        if (helper.getEnv().remove(envName) == null) {
            handler.error(EINVAL, "unsetenv", envName);
            return -1;
        }
        
        return 0;
    }
    
    public int utimes(String path, long[] atimeval, long[] mtimeval) {
        long mtimeMillis;
        if (mtimeval != null) {
            assert mtimeval.length == 2;
            mtimeMillis = (mtimeval[0] * 1000) + (mtimeval[1] / 1000);
        } else {
            mtimeMillis = System.currentTimeMillis();
        }
        new File(path).setLastModified(mtimeMillis);
        return 0;
    }

    public int utimes(String path, Pointer times) {
        return unimplementedInt("utimes");
    }

    public int futimes(int fd, long[] atimeval, long[] mtimeval) {
        handler.unimplementedError("futimes");
        return unimplementedInt("futimes");
    }

    public int lutimes(String path, long[] atimeval, long[] mtimeval) {
        handler.unimplementedError("lutimes");
        return unimplementedInt("lutimes");
    }

    public int utimensat(int dirfd, String path, long[] atimespec, long[] mtimespec, int flag) {
        long mtimeMillis;
        if (mtimespec != null) {
            assert mtimespec.length == 2;
            mtimeMillis = (mtimespec[0] * 1000) + (mtimespec[1] / 1000000);
        } else {
            mtimeMillis = System.currentTimeMillis();
        }
        new File(path).setLastModified(mtimeMillis);
        return 0;
    }

    public int utimensat(int dirfd, String path, Pointer times, int flag) {
        return unimplementedInt("utimensat");
    }

    public int futimens(int fd, long[] atimespec, long[] mtimespec) {
        handler.unimplementedError("futimens");
        return unimplementedInt("futimens");
    }

    public int futimens(int fd, Pointer times) {
        handler.unimplementedError("futimens");
        return unimplementedInt("futimens");
    }

    public int wait(int[] status) {
        return unimplementedInt("wait");
    }
    
    public int waitpid(int pid, int[] status, int flags) {
        return unimplementedInt("waitpid");
    }

    public int waitpid(long pid, int[] status, int flags) {
        return unimplementedInt("waitpid");
    }

    public int getpriority(int which, int who) {
        return unimplementedInt("getpriority");
    }
    
    public int setpriority(int which, int who, int prio) {
        return unimplementedInt("setpriority");
    }

    public long posix_spawnp(String path, Collection fileActions, Collection argv, Collection envp) {
        return unimplementedInt("posix_spawnp");
    }

    public long posix_spawnp(String path, Collection fileActions,
                             Collection spawnAttributes,
                             Collection argv, Collection envp) {
        return unimplementedInt("posix_spawnp");
    }

    public int errno() {
        return JavaLibCHelper.errno();
    }

    public void errno(int value) {
        JavaLibCHelper.errno(value);
    }
    
    public int chdir(String path) {
        return JavaLibCHelper.chdir(path);
    }

    public boolean isNative() {
        return false;
    }

    public LibC libc() {
        return null;
    }

    private int unimplementedInt(String message) {
        handler.unimplementedError(message);

        return -1;
    }

    public long sysconf(Sysconf name) {
        switch (name) {
            case _SC_CLK_TCK:
                return JavaTimes.HZ;

            default:
                errno(Errno.EOPNOTSUPP.intValue());
                return -1;
        }
    }

    public Times times() {
        return new JavaTimes();
    }
    
    public int flock(int fd, int mode) {
        return unimplementedInt("flock");
    }

    public int dup(int fd) {
        return unimplementedInt("dup");
    }

    public int dup2(int oldFd, int newFd) {
        return unimplementedInt("dup2");
    }

    public int fcntlInt(int fd, Fcntl fcntlConst, int arg) {
        return unimplementedInt("fcntl");
    }

    public int fcntl(int fd, Fcntl fcntlConst) {
        return unimplementedInt("fcntl");
    }

    public int fcntl(int fd, Fcntl fcntlConst, int... arg) {
        return unimplementedInt("fcntl");
    }

    public int access(CharSequence path, int amode) {
        handler.unimplementedError("access");

        return -1;
    }

    public int close(int fd) {
        return unimplementedInt("close");
    }

    public int unlink(CharSequence path) {
        handler.unimplementedError("unlink");

        return -1;
    }

    public int open(CharSequence path, int flags, int perm) {
        handler.unimplementedError("open");

        return -1;
    }


    public long read(int fd, byte[] buf, long n) {
        handler.unimplementedError("read");

        return -1;
    }
    public long write(int fd, byte[] buf, long n) {
        handler.unimplementedError("write");

        return -1;
    }
    public long read(int fd, ByteBuffer buf, long n) {
        handler.unimplementedError("read");

        return -1;
    }
    public long write(int fd, ByteBuffer buf, long n) {
        handler.unimplementedError("write");

        return -1;
    }
    public long pread(int fd, byte[] buf, long n, long offset) {
        handler.unimplementedError("pread");

        return -1;
    }
    public long pwrite(int fd, byte[] buf, long n, long offset) {
        handler.unimplementedError("pwrite");

        return -1;
    }
    public long pread(int fd, ByteBuffer buf, long n, long offset) {
        handler.unimplementedError("pread");

        return -1;
    }
    public long pwrite(int fd, ByteBuffer buf, long n, long offset) {
        handler.unimplementedError("pwrite");

        return -1;
    }

    public int read(int fd, byte[] buf, int n) {
        handler.unimplementedError("read");

        return -1;
    }
    public int write(int fd, byte[] buf, int n) {
        handler.unimplementedError("write");

        return -1;
    }
    public int read(int fd, ByteBuffer buf, int n) {
        handler.unimplementedError("read");

        return -1;
    }
    public int write(int fd, ByteBuffer buf, int n) {
        handler.unimplementedError("write");

        return -1;
    }
    public int pread(int fd, byte[] buf, int n, int offset) {
        handler.unimplementedError("pread");

        return -1;
    }
    public int pwrite(int fd, byte[] buf, int n, int offset) {
        handler.unimplementedError("pwrite");

        return -1;
    }
    public int pread(int fd, ByteBuffer buf, int n, int offset) {
        handler.unimplementedError("pread");

        return -1;
    }
    public int pwrite(int fd, ByteBuffer buf, int n, int offset) {
        handler.unimplementedError("pwrite");

        return -1;
    }

    public int lseek(int fd, long offset, int whence) {
        handler.unimplementedError("lseek");

        return -1;
    }

    public long lseekLong(int fd, long offset, int whence) {
        handler.unimplementedError("lseek");

        return -1;
    }

    public int pipe(int[] fds) {
        handler.unimplementedError("pipe");

        return -1;
    }

    public int socketpair(int domain, int type, int protocol, int[] fds) {
        handler.unimplementedError("socketpair");

        return -1;
    }

    public int sendmsg(int socket, MsgHdr message, int flags) {
        handler.unimplementedError("sendmsg");

        return -1;
    }

    public int recvmsg(int socket, MsgHdr message, int flags) {
        handler.unimplementedError("recvmsg");

        return -1;
    }

    public int truncate(CharSequence path, long length) {
        handler.unimplementedError("truncate");

        return -1;
    }

    public int ftruncate(int fd, long offset) {
        handler.unimplementedError("ftruncate");

        return -1;
    }

    public int rename(CharSequence oldName, CharSequence newName) {
        // Very basic support.  This might not work well with rename's semantics regarding symlinks.

        File oldFile = new File(oldName.toString());
        File newFile = new File(newName.toString());

        if (oldFile.renameTo(newFile)) {
            return 0;
        }

        return -1;
    }

    public String getcwd() {
        return System.getProperty("user.dir");
    }

    public int fsync(int fd) {
        handler.unimplementedError("fsync");
        return unimplementedInt("fsync not available for Java");
    }

    public int fdatasync(int fd) {
        handler.unimplementedError("fdatasync");
        return unimplementedInt("fdatasync not available for Java");
    }

    public int mkfifo(String filename, int mode) {
        handler.unimplementedError("mkfifo");
        return unimplementedInt("mkfifo not available for Java");
    }

    public int daemon(int nochdir, int noclose) {
        handler.unimplementedError("daemon");
        return unimplementedInt("daemon not available for Java");
    }

    public long[] getgroups() {
        handler.unimplementedError("getgroups");
        return null;
    }

    public int getgroups(int size, int[] groups) {
        handler.unimplementedError("getgroups");
        return unimplementedInt("getgroups not available for Java");
    }

    public String nl_langinfo(int item) {
        handler.unimplementedError("nl_langinfo");
        return null;
    }

    public String setlocale(int category, String locale) {
        handler.unimplementedError("setlocale");
        return null;
    }

    @Override
    public String strerror(int code) {
        handler.unimplementedError("strerror");
        return null;
    }

    public Timeval allocateTimeval() {
        handler.unimplementedError("allocateTimeval");
        return null;
    }

    @Override
    public int gettimeofday(Timeval tv) {
        handler.unimplementedError("gettimeofday");
        return -1;
    }

    static final class LoginInfo {
        public static final int UID = IDHelper.getInt("-u");
        public static final int GID = IDHelper.getInt("-g");
        public static final String USERNAME = IDHelper.getString("-un");
    }
    private static final class IDHelper {
        private static final String ID_CMD = Platform.IS_SOLARIS ? "/usr/xpg4/bin/id" : "/usr/bin/id";
        private static final int NOBODY = Platform.IS_WINDOWS ? 0 : Short.MAX_VALUE;
        public static int getInt(String option) {
            try {
                Process p = Runtime.getRuntime().exec(new String[] { ID_CMD, option });
                BufferedReader r = new BufferedReader(new InputStreamReader(p.getInputStream()));
                return Integer.parseInt(r.readLine());
            } catch (IOException ex) {
                return NOBODY;
            } catch (NumberFormatException ex) {
                return NOBODY;
            } catch (SecurityException ex) {
                return NOBODY;
            }
        }
        public static String getString(String option) {
            try {
                Process p = Runtime.getRuntime().exec(new String[] { ID_CMD, option });
                BufferedReader r = new BufferedReader(new InputStreamReader(p.getInputStream()));
                return r.readLine();
            } catch (IOException ex) {
                return null;
            }
        }
    }
    private static final class FakePasswd implements Passwd {

        public String getLoginName() {
            return LoginInfo.USERNAME;
        }

        public String getPassword() {
            return "";
        }

        public long getUID() {
            return LoginInfo.UID;
        }

        public long getGID() {
            return LoginInfo.GID;
        }

        public int getPasswdChangeTime() {
            return 0;
        }

        public String getAccessClass() {
            return "";
        }

        public String getGECOS() {
            return getLoginName();
        }

        public String getHome() {
            return "/";
        }

        public String getShell() {
            return "/bin/sh";
        }

        public int getExpire() {
            return ~0;
        }

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy