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

org.jclarion.clarion.ClarionFile Maven / Gradle / Ivy

/**
 * Copyright 2010, by Andrew Barnham
 *
 * The contents of this file are subject to
 * GNU Lesser General Public License (LGPL), v.3
 * http://www.gnu.org/licenses/lgpl.txt
 * 
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied.
 */
package org.jclarion.clarion;

import java.util.ArrayList;
import java.util.List;
import java.util.StringTokenizer;

//import org.jclarion.Params;
import org.jclarion.clarion.memory.CMem;
import org.jclarion.clarion.primative.AbstractStateFactory;
import org.jclarion.clarion.primative.AbstractStateGetter;
import org.jclarion.clarion.primative.GlobalStateGetter;
import org.jclarion.clarion.primative.ThreadStateGetter;
import org.jclarion.clarion.runtime.CErrorImpl;
import org.jclarion.clarion.runtime.CMemory;
import org.jclarion.clarion.util.FileState;
import org.jclarion.clarion.constants.*;
import org.jclarion.clarion.hooks.FilecallbackParams;
import org.jclarion.clarion.hooks.Filecallbackinterface;

/**
 *  Provide abstract model of a File entity in clarion.
 *  
 * @author barney
 *
 */
public abstract class ClarionFile extends ClarionGroup
{
    private List keys=new ArrayList();

    public int getSQLType(ClarionObject o)
    {
        FileState fs=  getFileState();
        int indx = fs.getFieldIndex(o);
        return fs.global.types[indx];
    }

    private static class StateFactory extends AbstractStateFactory
    {
        @Override
        public FileState cloneState(FileState base) {
            return new FileState(base);
        }

        @Override
        public FileState createState() {
            return new FileState();
        }
    }
    private static StateFactory factory=new StateFactory();
    
    private AbstractStateGetter state=new GlobalStateGetter(factory);

    @Override
    public void initThread()
    {
        if (!state.isThreaded()) state=new ThreadStateGetter(factory,state);
        super.initThread();
    }
    
    public void setCreate()
    {
        setProperty(Prop.CREATE,true);
    }
    
    /**
     * Add sort key
     * 
     * @param key
     */
    public void addKey(ClarionKey key) 
    {
        keys.add(key);
        key.setFile(this);
    }

    /**
     * Remove a sort key. Used by views
     * 
     * @param key
     */
    public void removeKey(ClarionKey key) 
    {
        keys.remove(key);
        key.setFile(null);
    }
    
    public void setNoMemo()
    {
    }
    
    public int getBytes()
    {
        return 0;
    }

    public void removeKeys() 
    {
        keys.clear();
    }
    
    
    /**
     * Specify owner of the file. For SQL drivers it indicates where 
     * want to connect to 
     * 
     * @param source also known as owner
     * @return
     */
    public ClarionFile setSource(ClarionString source)
    {
        setProperty(Prop.OWNER,source);
        return this;
    }

    /**
     * Set name of file to read. i.e. file name for filesystem access
     * and table name ofr SQL access
     *  
     * @param name
     * @return
     */
    public ClarionFile setName(String name)
    {
        setProperty(Prop.NAME,name);
        return this;
    }

    public ClarionFile setName(ClarionString name)
    {
        setProperty(Prop.NAME,name);
        return this;
    }


    /**
     * Open the file in read/write local but deny write others mode
     */
    public void open()
    {
        open(0x22);
    }

    /**
     * Set output write caching on a file. By default it is disabled. Only
     * applies for filesystem type files.
     */
    public abstract void stream();

    /**
     *  Delete a file.
     */
    public abstract void remove();

    /**
     *  Flush file buffers. Only applies for filesystem type files.
     */
    public abstract void flush();

    /**
     *  Open file in particular mode. Supported modes are:
     *  
     *  User access:
     *      x00 = read only
     *      x01 = write only
     *      x02 = read/write
     *      
     *  Remote access:
     *      x00 = any
     *      x10 = deny all    
     *      x20 = deny write    
     *      x30 = deny read    
     *      x40 = deny nothing    
     *      
     * Open default is x22 = Read/Write user but deny write remote.     
     *      
     * @param mode
     */
    public abstract void open(int mode);

    /**
     *  Close a file
     */
    public abstract void close();

    /**
     *  Rebuild key files
     */
    public abstract void build();

    /**
     *  Get next record in current sequence
     */
    public abstract void next();
    
    /**
     *  Get prior record in current sequence
     */
    public abstract void previous();

    /**
     * Rewrite current record
     */
    public abstract void put();

    /**
     *  Delete current record
     *  
     */
    public abstract void delete();

    /**
     * Copy file
     */
    public abstract void copyTo(String target);
    
    /**
     *  Create new file. If file already exists it will be overwritten. 
     *  this call open after.
     */
    public abstract void create();

    /**
     *  Lock file
     */
    public abstract void lock();
    
    /**
     * Enable optimistic locking. 
     * 
     * On next get,next,previous or reget. make a note of contents of record.
     * 
     * On subsequent put, fail operation if the record has changed in anyway
     * error code for failure is 89.
     * 
     * A second get,next,previous or reget will cancel the watch.
     */
    public abstract void watch();
    
    /**
     * Set record paging. We will never bother with this one but will make
     * it abstract anyway.
     * 
     * @param pagesize
     * @param behind
     * @param ahead
     * @param timeout
     */
    public abstract void buffer(Integer pagesize,Integer behind,Integer ahead,Integer timeout);

    /**
     * Unlock previously locked file
     */
    public abstract void unlock();

    /**
     * Get record based on particular key
     * @param akey
     */
    public abstract void get(ClarionKey akey);
    
    /**
     * insert new record based on contents of record buffer
     */
    public abstract void add();

    /**
     * insert new record writing only fixed # of bytes. Mainly for ascii
     * and binary file drivers
     */
    public abstract void add(int size);

    /**
     * Return number of records in file
     * @return
     */
    public abstract int records();
    
    /**
     * retrieve record based on specified pointer position - previously
     * returned by pointer command.
     * 
     * @param pointer - the position to get
     * @param len - the number of bytes to read, for dos drivers etc
     * 
     * @return
     */
    public abstract void get(ClarionString pointer,Integer len);

    public abstract void get(ClarionNumber position, Integer len);
    
    
    /**
     *  Return true if at end of file
     * @return
     */
    public abstract boolean eof();

    /**
     *  Retrun true if at beginning of file
     * @return
     */
    public abstract boolean bof();

    /**
     *  Set sequentual record processing
     */
    public final void set()
    {
        set(null);
    }

    /**
     *  Set key based record processing
     */
    public abstract void set(ClarionKey key);
    
    
    /**
     * Set sequentual record processing starting at position
     *  
     * @param position position - as returned by position method
     */
    public abstract void set(int position);

    /**
     *  Send special command to the file driver 
     * @param operator
     */
    public abstract void send(String operator);
    
    protected byte[] decodeByteArraySend(String value)
    {
        StringTokenizer tok = new StringTokenizer(value," ,");
        byte result[]=new byte[Integer.parseInt(tok.nextToken())];
        for (int scan=0;scankeys.size()) return null;
        return keys.get(indx-1);
    }

    public List getKeys()
    {
        return keys;
    }
    
    public ClarionKey getPrimaryKey()
    {
        for (ClarionKey key : keys ) {
            if (key.isProperty(Prop.PRIMARY)) return key;
        }
        return null;
    }
    
    /** 
     * Save state of file to be restored later. Saving
     *  Current file state, open etc
     *  Contents of record buffer
     *  Watch status
     *  Variable change status
     *  Null variable states
     *  Sequentual processing position
     */
    public abstract int getState();
    
    /**
     *  Free memory held by stored state but do not restore state
     * @param state
     */
    public abstract void freeState(int state);

    /**
     * Restore state. Memory is not freed
     * @param state
     */
    public abstract void restoreState(int state);

    /**
     * Get encoding of status of nulls used by SQL files
     * @return
     */
    public abstract ClarionString getNulls();

    /**
     *  Restore encoding of status of nulls used by SQL files
     * @param nulls
     */
    public abstract void setNulls(ClarionString nulls);

    /**
     * return file name
     * 
     * @return
     */
    public String getName()
    {
        throw new RuntimeException("Not yet implemented");
    }
    

    /** 
     * initialise transaction processing
     * 
     * @param timeout
     */
    public static void logout(int timeout,ClarionFile... files)
    {
        //throw new RuntimeException("Not yet implemented");
    }

    /**
     * rollback transaction changes
     */
    public static void rollback()
    {
        //throw new RuntimeException("Not yet implemented");
    }

    /**
     * Commit transaction changes
     */
    public static void commit()
    {
        //throw new RuntimeException("Not yet implemented");
    }
    
    public FileState getFileState()
    {
        FileState fs = state.get();
        if (fs.fields==null) {
            initFileState(fs);
        }
        return fs;
    }
    
    private void initFileState(final FileState fs)
    {
        fs.changed=new boolean[getVariableCount()];
        fs.isnull=new boolean[getVariableCount()];
        fs.fields=new ClarionObject[getVariableCount()];
        fs.listeners=new ClarionMemoryChangeListener[getVariableCount()];
        for (int scan=0;scan fileCallbacks=new ArrayList();

    public void registerCallback(Filecallbackinterface inf,boolean junk)
    {
        fileCallbacks.add(inf);
    }
    
    public void deregisterCallback(Filecallbackinterface inf)
    {
        fileCallbacks.remove(inf);
    }
    
    public void functionCalled(int code,FilecallbackParams params,String ecode,String emsg)
    {
        for (Filecallbackinterface cb : fileCallbacks ) {
            cb.functionCalled(
                    Clarion.newNumber(code),
                    params,
                    Clarion.newString(ecode),
                    Clarion.newString(emsg));
        }
    }
    
    public void functionDone(int code,FilecallbackParams params,String ecode,String emsg)
    {
        for (Filecallbackinterface cb : fileCallbacks ) {
            cb.functionDone(
                    Clarion.newNumber(code),
                    params,
                    Clarion.newString(ecode),
                    Clarion.newString(emsg));
        }
    }

    public boolean testOpen()
    {
        CErrorImpl.getInstance().clearError();
        if (getFileState().global.openCount==0) {
            CErrorImpl.getInstance().setError(37,"File Not Open");
            return false;
        }
        return true;
    }

    public boolean duplicateCheck(ClarionKey clarionKey) {
        throw new RuntimeException("Not yet implemented");
    }

    public static boolean isPositionMatches(ClarionString string,ClarionObject[] scanFields) 
    {
        if (scanFields==null) return false;
        for (int scan=0;scan




© 2015 - 2025 Weber Informatics LLC | Privacy Policy