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

IceUtilInternal.FileLock Maven / Gradle / Ivy

Go to download

Ice is a comprehensive RPC framework that helps you build distributed applications with minimal effort using familiar object-oriented idioms

There is a newer version: 3.7.10
Show newest version
// **********************************************************************
//
// Copyright (c) 2003-2017 ZeroC, Inc. All rights reserved.
//
// This copy of Ice is licensed to you under the terms described in the
// ICE_LICENSE file included in this distribution.
//
// **********************************************************************

package IceUtilInternal;

import java.io.File;
import java.io.RandomAccessFile;
import java.nio.channels.FileChannel;
import java.lang.reflect.Method;

public final class FileLock
{
    //
    // The constructor opens the given file (eventually creating it)
    // and acquires a lock on the file or throws FileLockException if
    // the file couldn't be locked.
    //
    // If the lock can be acquired, the Java VM name is written to the
    // file.
    //
    public FileLock(String path)
    {
        _file = new File(path);

        FileChannel channel;
        try
        {
            _randFile = new RandomAccessFile(_file, "rw");
            channel = _randFile.getChannel();
        }
        catch(java.io.FileNotFoundException e)
        {
            throw new IceUtil.FileLockException(path);
        }

        java.nio.channels.FileLock lock;
        try
        {
            lock = channel.tryLock();
        }
        catch(Exception ex)
        {
            throw new IceUtil.FileLockException(path, ex);
        }

        if(lock == null)
        {
            throw new IceUtil.FileLockException(path);
        }

        //
        // In Windows we don't write the process pid to the file, as is  not posible
        // to read the file from other process while it is locked here.
        //
        if(!System.getProperty("os.name").startsWith("Windows"))
        {
            try
            {
                //
                // Java doesn't provide get pid operation. This code
                // writes the Java VM name instead of the pid.
                //
                // The output is JVM dependent. With the Sun
                // implementation it's `pid@hostname'
                //
                try
                {
                    //
                    // We access java.lang.management classes using reflection
                    // because these classes are not available with Android
                    // Dalvik JVM.
                    //
                    if(!System.getProperty("java.vm.name").startsWith("Dalvik"))
                    {
                        Class fC = IceInternal.Util.findClass("java.lang.management.ManagementFactory", null);
                        Class mC = IceInternal.Util.findClass("java.lang.management.RuntimeMXBean", null);

                        Method getRuntimeMXBean = fC.getDeclaredMethod("getRuntimeMXBean", (Class[])null);

                        Method getName = mC.getDeclaredMethod("getName", (Class[])null);

                        Object mxBean = getRuntimeMXBean.invoke(null);
                        _randFile.writeUTF((String)getName.invoke(mxBean));
                    }
                    else
                    {
                        //
                        // In Android with Dalvik we can use android.os.Process to get the
                        // process pid. That is done using reflection because it is specific
                        // for Android Dalvik VM.
                        //
                        Class pC = IceInternal.Util.findClass("android.os.Process", null);
                        Method myPid = pC.getDeclaredMethod("myPid", (Class[])null);
                        _randFile.writeUTF(((Integer)myPid.invoke(null)).toString());
                    }
                }
                catch(NoSuchMethodException ex)
                {
                }
                catch(IllegalAccessException ex)
                {
                }
                catch(java.lang.reflect.InvocationTargetException ex)
                {
                }
                //
                // Don't close _randFile here or the lock will be released. It is called
                // during release see comments there.
                //
            }
            catch(java.io.IOException ex)
            {
                release();
                throw new IceUtil.FileLockException(path, ex);
            }
        }
    }

    //
    // Remove the lock if it is owned by the class instance.
    //
    public void release()
    {
        if(System.getProperty("os.name").startsWith("Windows"))
        {
            if(_randFile != null)
            {
                //
                // In Windows we need to close the file handler before
                // we try to delete the file. Note that the call to close
                // also release the file lock.
                //
                try
                {
                    _randFile.close();
                }
                catch(java.io.IOException ex)
                {
                }
                _randFile = null;
            }
        }
        //
        // on UNIX the call to delete remove the file and that
        // release the lock.
        //
        // In Windows the call to delete will success if at that point
        // the file has not been opened by any process.
        //
        if(_file != null)
        {
            _file.delete();
            _file = null;
        }
    }

    //
    // We need to keep these refrences to delete the lock file during release
    //
    private File _file;
    private RandomAccessFile _randFile;

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy