import java.io.* ;
import java.text.SimpleDateFormat ;
import java.util.* ;
class Expense_8_puzzle implements Comparable< Expense_8_puzzle> {
int [ ] [ ] mat;
int [ ] movablePlaceHolder;
int cost;
int depth;
Expense_8_puzzle predecessor;
static final Map
< String ,
int [ ] > movableActions
= new HashMap
<> ( ) ;
static {
movableActions.put ( "DOWN" , new int [ ] { - 1 , 0 } ) ;
movableActions.put ( "UP" , new int [ ] { 1 , 0 } ) ;
movableActions.put ( "RIGHT" , new int [ ] { 0 , - 1 } ) ;
movableActions.put ( "LEFT" , new int [ ] { 0 , 1 } ) ;
}
Expense_8_puzzle
( int [ ] [ ] mat,
int [ ] movablePlaceHolder,
int cost,
int depth, Expense_8_puzzle predecessor,
String action
) { this .mat = new int [ 3 ] [ 3 ] ;
for ( int i = 0 ; i < 3 ; i++ ) {
this .
mat [ i
] = Arrays .
copyOf ( mat
[ i
] , mat
[ i
] .
length ) ; }
this .
movablePlaceHolder = Arrays .
copyOf ( movablePlaceHolder, movablePlaceHolder.
length ) ; this .cost = cost;
this .depth = depth;
this .predecessor = predecessor;
this .action = action;
}
@Override
public int compareTo( Expense_8_puzzle other) {
return Integer .
compare ( this .
cost , other.
cost ) ; }
// Method to parse input file and create the problem state
int [ ] [ ] mat = new int [ 3 ] [ 3 ] ;
int [ ] movablePlaceHolder = new int [ 2 ] ;
for ( int i = 0 ; i < 3 ; i++ ) {
line = br.readLine ( ) ;
String [ ] tokens
= line.
split ( " " ) ; for ( int j = 0 ; j < tokens.length ; j++ ) {
mat
[ i
] [ j
] = Integer .
parseInt ( tokens
[ j
] ) ; if ( mat[ i] [ j] == 0 ) {
movablePlaceHolder[ 0 ] = i;
movablePlaceHolder[ 1 ] = j;
}
}
}
br.close ( ) ;
return new Expense_8_puzzle( mat, movablePlaceHolder, 0 , 0 , null , null ) ;
}
// Expansions for the current state (children nodes)
List< Expense_8_puzzle> getExpansions( ) {
List< Expense_8_puzzle> expansions = new ArrayList<> ( ) ;
int x = movablePlaceHolder[ 0 ] , y = movablePlaceHolder[ 1 ] ;
for ( Map .
Entry < String ,
int [ ] > entry
: movableActions.
entrySet ( ) ) { String action
= entry.
getKey ( ) ; int [ ] dir = entry.getValue ( ) ;
int tempA = x + dir[ 0 ] , tempB = y + dir[ 1 ] ;
// Valid grid boundaries
if ( tempA >= 0 && tempA < 3 && tempB >= 0 && tempB < 3 ) {
int [ ] [ ] tempMat = new int [ 3 ] [ 3 ] ;
for ( int i = 0 ; i < 3 ; i++ ) {
tempMat
[ i
] = Arrays .
copyOf ( mat
[ i
] , mat
[ i
] .
length ) ; }
tempMat[ x] [ y] = tempMat[ tempA] [ tempB] ;
tempMat[ tempA] [ tempB] = 0 ;
int tileValue = tempMat[ x] [ y] ;
expansions.add ( new Expense_8_puzzle( tempMat, new int [ ] { tempA, tempB} , this .cost + tileValue, this .depth + 1 , this , action) ) ;
}
}
return expansions;
}
// Trace the path from current to root (predecessors)
List< String> parentChainGenerator( ) {
List< String> path = new ArrayList<> ( ) ;
Expense_8_puzzle state = this ;
while ( state.predecessor != null ) {
path.add ( "Move " + state.mat [ state.predecessor .movablePlaceHolder [ 0 ] ] [ state.predecessor .movablePlaceHolder [ 1 ] ] + " " + state.action ) ;
state = state.predecessor ;
}
return path; // No need to reverse. It prints from current to root.
}
// Heuristic function
static int estimate_heuristicValues( Expense_8_puzzle state, Expense_8_puzzle goal) {
int cost = 0 ;
for ( int i = 0 ; i < 3 ; i++ ) {
for ( int j = 0 ; j < 3 ; j++ ) {
int tile = state.mat [ i] [ j] ;
if ( tile != 0 ) {
int goalI = ( tile - 1 ) / 3 ;
int goalJ = ( tile - 1 ) % 3 ;
int moves
= Math .
abs ( i
- goalI
) + Math .
abs ( j
- goalJ
) ; cost += moves * tile;
}
}
}
return cost;
}
// Parent chain from current state to root (no reversal)
static List< Expense_8_puzzle> getParentChain( Expense_8_puzzle state) {
List< Expense_8_puzzle> chain = new ArrayList<> ( ) ;
while ( state != null ) {
chain.add ( state) ; // Start from current state
state = state.predecessor ;
}
return chain;
}
// Log state and its predecessor chain
static void dumpLogger
( Expense_8_puzzle presentPuzzleMat, List
< Expense_8_puzzle
> successors, Set
< String
> closedSet, Collection
< Expense_8_puzzle
> frontier,
int NPop,
String logFile, Expense_8_puzzle goalState,
String method
) throws IOException { if ( logFile != null ) {
if ( method.equals ( "a*" ) || method.equals ( "greedy" ) ) {
fileEditor.
write ( String .
format ( "Generating successors to < state = %s, action = {%s} g(n) = %d, d = %d, f(n) = %d, Parent = Pointer to %s >:\n " ,
Arrays .
deepToString ( presentPuzzleMat.
mat ) , presentPuzzleMat.
action == null ? "Start" : presentPuzzleMat.
action , presentPuzzleMat.
cost ,
presentPuzzleMat.depth , presentPuzzleMat.cost + estimate_heuristicValues( presentPuzzleMat, goalState) ,
presentPuzzleMat.
predecessor == null ? "None" : Arrays .
deepToString ( presentPuzzleMat.
predecessor .
mat ) ) ) ; } else {
fileEditor.
write ( String .
format ( "Generating successors to < state = %s, action = {%s} cost = %d, d = %d, Parent = Pointer to %s >:\n " ,
Arrays .
deepToString ( presentPuzzleMat.
mat ) , presentPuzzleMat.
action == null ? "Start" : presentPuzzleMat.
action , presentPuzzleMat.
cost ,
presentPuzzleMat.
depth , presentPuzzleMat.
predecessor == null ? "None" : Arrays .
deepToString ( presentPuzzleMat.
predecessor .
mat ) ) ) ; }
fileEditor.write ( "\t successors generated -> \n " ) ;
fileEditor.write ( "\t Closed: " + closedSet + "\n " ) ;
fileEditor.write ( "\t Fringe: [\n " ) ;
// Iterate over frontier and log the state and its predecessor chain
for ( Expense_8_puzzle state : frontier) {
List< Expense_8_puzzle> predecessorChain = getParentChain( state) ; // Correct predecessor chain
// Print chain from current state to root
StringBuilder predecessorChainStr = new StringBuilder( ) ;
for ( int i = 0 ; i < predecessorChain.size ( ) ; i++ ) { // Keep order as-is
Expense_8_puzzle p = predecessorChain.get ( i) ;
predecessorChainStr.
append ( String .
format ( "< state = %s, action = {Move %s}, g(n) = %d, d = %d, f(n) = %d >" ,
Arrays .
deepToString ( p.
mat ) , p.
action , p.
cost , p.
depth , method.
equals ( "a*" ) || method.
equals ( "greedy" ) ? ( p.
cost + estimate_heuristicValues
( p, goalState
) ) : p.
cost ) ) ; if ( i != predecessorChain.size ( ) - 1 ) { // Append arrow unless last state
predecessorChainStr.append ( " -> " ) ;
}
}
fileEditor.write ( "\t \t " + predecessorChainStr.toString ( ) + "\n " ) ;
}
fileEditor.write ( "\t ]\n " ) ;
fileEditor.close ( ) ;
}
}
// DLS (Depth-Limited Search)
static Expense_8_puzzle dls
( Expense_8_puzzle start, Expense_8_puzzle goal,
int DL,
boolean logBoolean,
String logFile
) throws IOException { Stack< Expense_8_puzzle> frontier = new Stack<> ( ) ;
Set< String> closedSet = new HashSet<> ( ) ;
frontier.push ( start) ;
int NPop = 0 , NGen = 1 , NExp = 0 , MFSize = 0 ;
while ( ! frontier.isEmpty ( ) ) {
MFSize
= Math .
max ( MFSize, frontier.
size ( ) ) ; Expense_8_puzzle presentPuzzleMat = frontier.pop ( ) ;
NPop++;
if ( presentPuzzleMat.depth > DL) {
continue ;
}
if ( Arrays .
deepEquals ( presentPuzzleMat.
mat , goal.
mat ) ) { System .
out .
println ( "Nodes Expanded: " + NExp
) ; System .
out .
println ( "Nodes Generated: " + NGen
) ; System .
out .
println ( "Nodes Popped: " + NPop
) ; System .
out .
println ( "Max Fringe Size: " + MFSize
) ; return presentPuzzleMat;
}
if ( closedSet.contains ( matStr) ) {
continue ;
}
closedSet.add ( matStr) ;
NExp++;
List< Expense_8_puzzle> successors = presentPuzzleMat.getExpansions ( ) ;
for ( Expense_8_puzzle neighbor : successors) {
if ( neighbor.depth <= DL) {
frontier.push ( neighbor) ;
NGen++;
}
}
if ( logBoolean) {
dumpLogger( presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "dls" ) ;
}
}
return null ;
}
// BFS
static Expense_8_puzzle bfs
( Expense_8_puzzle start, Expense_8_puzzle goal,
boolean logBoolean,
String logFile
) throws IOException { Queue< Expense_8_puzzle> frontier = new LinkedList<> ( ) ;
Set< String> closedSet = new HashSet<> ( ) ;
frontier.add ( start) ;
int NPop = 0 , NGen = 1 , NExp = 0 , MFSize = 0 ;
while ( ! frontier.isEmpty ( ) ) {
MFSize
= Math .
max ( MFSize, frontier.
size ( ) ) ; Expense_8_puzzle presentPuzzleMat = frontier.poll ( ) ;
NPop++;
if ( Arrays .
deepEquals ( presentPuzzleMat.
mat , goal.
mat ) ) { System .
out .
println ( "Nodes Expanded: " + NExp
) ; System .
out .
println ( "Nodes Generated: " + NGen
) ; System .
out .
println ( "Nodes Popped: " + NPop
) ; System .
out .
println ( "Max Fringe Size: " + MFSize
) ; return presentPuzzleMat;
}
if ( closedSet.contains ( matStr) ) {
continue ;
}
closedSet.add ( matStr) ;
NExp++;
List< Expense_8_puzzle> successors = presentPuzzleMat.getExpansions ( ) ;
for ( Expense_8_puzzle neighbor : successors) {
frontier.add ( neighbor) ;
NGen++;
}
if ( logBoolean) {
dumpLogger( presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "bfs" ) ;
}
}
return null ;
}
// IDS
static Expense_8_puzzle ids
( Expense_8_puzzle start, Expense_8_puzzle goal,
boolean logBoolean,
String logFile
) throws IOException { int DL = 0 ;
while ( true ) {
Expense_8_puzzle result = dls( start, goal, DL, logBoolean, logFile) ;
if ( result != null ) {
return result;
}
DL++;
}
}
// UCS
static Expense_8_puzzle ucs
( Expense_8_puzzle start, Expense_8_puzzle goal,
boolean logBoolean,
String logFile
) throws IOException { PriorityQueue< Expense_8_puzzle> frontier = new PriorityQueue<> ( ) ;
Set< String> closedSet = new HashSet<> ( ) ;
frontier.add ( start) ;
int NPop = 0 , NGen = 1 , NExp = 0 , MFSize = 0 ;
while ( ! frontier.isEmpty ( ) ) {
MFSize
= Math .
max ( MFSize, frontier.
size ( ) ) ; Expense_8_puzzle presentPuzzleMat = frontier.poll ( ) ;
NPop++;
if ( Arrays .
deepEquals ( presentPuzzleMat.
mat , goal.
mat ) ) { System .
out .
println ( "Nodes Expanded: " + NExp
) ; System .
out .
println ( "Nodes Generated: " + NGen
) ; System .
out .
println ( "Nodes Popped: " + NPop
) ; System .
out .
println ( "Max Fringe Size: " + MFSize
) ; return presentPuzzleMat;
}
if ( closedSet.contains ( matStr) ) {
continue ;
}
closedSet.add ( matStr) ;
NExp++;
List< Expense_8_puzzle> successors = presentPuzzleMat.getExpansions ( ) ;
for ( Expense_8_puzzle neighbor : successors) {
frontier.add ( neighbor) ;
NGen++;
}
if ( logBoolean) {
dumpLogger( presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "ucs" ) ;
}
}
return null ;
}
// Greedy
static Expense_8_puzzle greedy
( Expense_8_puzzle start, Expense_8_puzzle goal,
boolean logBoolean,
String logFile
) throws IOException { PriorityQueue
< Expense_8_puzzle
> frontier
= new PriorityQueue
<> ( Comparator .
comparingInt ( s
-> estimate_heuristicValues
( s, goal
) ) ) ; Set< String> closedSet = new HashSet<> ( ) ;
frontier.add ( start) ;
int NPop = 0 , NGen = 1 , NExp = 0 , MFSize = 0 ;
while ( ! frontier.isEmpty ( ) ) {
MFSize
= Math .
max ( MFSize, frontier.
size ( ) ) ; Expense_8_puzzle presentPuzzleMat = frontier.poll ( ) ;
NPop++;
if ( Arrays .
deepEquals ( presentPuzzleMat.
mat , goal.
mat ) ) { System .
out .
println ( "Nodes Expanded: " + NExp
) ; System .
out .
println ( "Nodes Generated: " + NGen
) ; System .
out .
println ( "Nodes Popped: " + NPop
) ; System .
out .
println ( "Max Fringe Size: " + MFSize
) ; return presentPuzzleMat;
}
if ( closedSet.contains ( matStr) ) {
continue ;
}
closedSet.add ( matStr) ;
NExp++;
List< Expense_8_puzzle> successors = presentPuzzleMat.getExpansions ( ) ;
for ( Expense_8_puzzle neighbor : successors) {
frontier.add ( neighbor) ;
NGen++;
}
if ( logBoolean) {
dumpLogger( presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "greedy" ) ;
}
}
return null ;
}
// A*
static Expense_8_puzzle aStar
( Expense_8_puzzle start, Expense_8_puzzle goal,
boolean logBoolean,
String logFile
) throws IOException { PriorityQueue
< Expense_8_puzzle
> frontier
= new PriorityQueue
<> ( Comparator .
comparingInt ( s
-> s.
cost + estimate_heuristicValues
( s, goal
) ) ) ; Set< String> closedSet = new HashSet<> ( ) ;
frontier.add ( start) ;
int NPop = 0 , NGen = 1 , NExp = 0 , MFSize = 0 ;
while ( ! frontier.isEmpty ( ) ) {
MFSize
= Math .
max ( MFSize, frontier.
size ( ) ) ; Expense_8_puzzle presentPuzzleMat = frontier.poll ( ) ;
NPop++;
if ( Arrays .
deepEquals ( presentPuzzleMat.
mat , goal.
mat ) ) { System .
out .
println ( "Nodes Expanded: " + NExp
) ; System .
out .
println ( "Nodes Generated: " + NGen
) ; System .
out .
println ( "Nodes Popped: " + NPop
) ; System .
out .
println ( "Max Fringe Size: " + MFSize
) ; return presentPuzzleMat;
}
if ( closedSet.contains ( matStr) ) {
continue ;
}
closedSet.add ( matStr) ;
NExp++;
List< Expense_8_puzzle> successors = presentPuzzleMat.getExpansions ( ) ;
for ( Expense_8_puzzle neighbor : successors) {
frontier.add ( neighbor) ;
NGen++;
}
if ( logBoolean) {
dumpLogger( presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "a*" ) ;
}
}
return null ;
}
if ( args.length < 2 ) {
System .
out .
println ( "Usage: java Expense_8_puzzle <start-file> <goal-file> <method> <log-file>" ) ; return ;
}
boolean logBoolean = false ;
String logFile
= "traceDateTime.txt" ; // trace-Date-Time.txt
if ( args.length > 2 ) {
if ( args[ 2 ] .equals ( "true" ) || args[ 2 ] .equals ( "false" ) ) {
logBoolean
= Boolean .
parseBoolean ( args
[ 2 ] ) ; }
else {
method = args[ 2 ] ;
}
}
if ( args.length > 3 ) {
logBoolean
= Boolean .
parseBoolean ( args
[ 3 ] ) ; }
Expense_8_puzzle startState = Expense_8_puzzle.convertToMatFromFile ( startFile) ;
Expense_8_puzzle goalState = Expense_8_puzzle.convertToMatFromFile ( goalFile) ;
Expense_8_puzzle answer = null ;
switch ( method) {
case "bfs" :
answer = Expense_8_puzzle.bfs ( startState, goalState, logBoolean, logFile) ;
break ;
case "dfs" :
answer
= Expense_8_puzzle.
dls ( startState, goalState,
Integer .
MAX_VALUE , logBoolean, logFile
) ; break ;
case "dls" :
int DL = 10 ;
Scanner sc
= new Scanner
( System .
in ) ; System .
out .
println ( "Enter the depth limit: " ) ; String inputTaken
= sc.
nextLine ( ) ; if ( inputTaken.length ( ) > 0 && inputTaken.matches ( "[0-9]+" ) ) {
}
else {
System .
out .
println ( "Invalid input, using default depth limit of 10" ) ; }
answer = Expense_8_puzzle.dls ( startState, goalState, DL, logBoolean, logFile) ;
break ;
case "ids" :
answer = Expense_8_puzzle.ids ( startState, goalState, logBoolean, logFile) ;
break ;
case "ucs" :
answer = Expense_8_puzzle.ucs ( startState, goalState, logBoolean, logFile) ;
break ;
case "greedy" :
answer = Expense_8_puzzle.greedy ( startState, goalState, logBoolean, logFile) ;
break ;
case "a*" :
answer = Expense_8_puzzle.aStar ( startState, goalState, logBoolean, logFile) ;
break ;
default :
System .
out .
println ( "Unknown method!" ) ; return ;
}
if ( answer != null ) {
System .
out .
println ( "Solution Found at depth " + answer.
depth + " with cost of " + answer.
cost ) ; List< String> steps = answer.parentChainGenerator ( ) ;
System .
out .
println ( "\t " + step
) ; }
} else {
System .
out .
println ( "No answer found." ) ; }
}
}
import java.io.*;
import java.text.SimpleDateFormat;
import java.util.*;

class Expense_8_puzzle implements Comparable<Expense_8_puzzle> {
    int[][] mat;
    int[] movablePlaceHolder;
    int cost;
    int depth;
    Expense_8_puzzle predecessor;
    String action;

    static final Map<String, int[]> movableActions = new HashMap<>();

    static {
        movableActions.put("DOWN", new int[]{-1, 0});
        movableActions.put("UP", new int[]{1, 0});
        movableActions.put("RIGHT", new int[]{0, -1});
        movableActions.put("LEFT", new int[]{0, 1});
    }

    Expense_8_puzzle(int[][] mat, int[] movablePlaceHolder, int cost, int depth, Expense_8_puzzle predecessor, String action) {
        this.mat = new int[3][3];
        for (int i = 0; i < 3; i++) {
            this.mat[i] = Arrays.copyOf(mat[i], mat[i].length);
        }
        this.movablePlaceHolder = Arrays.copyOf(movablePlaceHolder, movablePlaceHolder.length);
        this.cost = cost;
        this.depth = depth;
        this.predecessor = predecessor;
        this.action = action;
    }

    @Override
    public int compareTo(Expense_8_puzzle other) {
        return Integer.compare(this.cost, other.cost);
    }

    // Method to parse input file and create the problem state
    static Expense_8_puzzle convertToMatFromFile(String fileName) throws IOException {
        BufferedReader br = new BufferedReader(new FileReader(fileName));
        String line;
        int[][] mat = new int[3][3];
        int[] movablePlaceHolder = new int[2];

        for (int i = 0; i < 3; i++) {
            line = br.readLine();
            String[] tokens = line.split(" ");
            for (int j = 0; j < tokens.length; j++) {
                mat[i][j] = Integer.parseInt(tokens[j]);
                if (mat[i][j] == 0) {
                    movablePlaceHolder[0] = i;
                    movablePlaceHolder[1] = j;
                }
            }
        }
        br.close();
        return new Expense_8_puzzle(mat, movablePlaceHolder, 0, 0, null, null);
    }

    // Expansions for the current state (children nodes)
    List<Expense_8_puzzle> getExpansions() {
        List<Expense_8_puzzle> expansions = new ArrayList<>();
        int x = movablePlaceHolder[0], y = movablePlaceHolder[1];

        for (Map.Entry<String, int[]> entry : movableActions.entrySet()) {
            String action = entry.getKey();
            int[] dir = entry.getValue();
            int tempA = x + dir[0], tempB = y + dir[1];

            // Valid grid boundaries
            if (tempA >= 0 && tempA < 3 && tempB >= 0 && tempB < 3) {
                int[][] tempMat = new int[3][3];
                for (int i = 0; i < 3; i++) {
                    tempMat[i] = Arrays.copyOf(mat[i], mat[i].length);
                }
                tempMat[x][y] = tempMat[tempA][tempB];
                tempMat[tempA][tempB] = 0;
                int tileValue = tempMat[x][y];
                expansions.add(new Expense_8_puzzle(tempMat, new int[]{tempA, tempB}, this.cost + tileValue, this.depth + 1, this, action));
            }
        }
        return expansions;
    }

    // Trace the path from current to root (predecessors)
    List<String> parentChainGenerator() {
        List<String> path = new ArrayList<>();
        Expense_8_puzzle state = this;

        while (state.predecessor != null) {
            path.add("Move " + state.mat[state.predecessor.movablePlaceHolder[0]][state.predecessor.movablePlaceHolder[1]] + " " + state.action);
            state = state.predecessor;
        }
        return path;  // No need to reverse. It prints from current to root.
    }

    // Heuristic function
    static int estimate_heuristicValues(Expense_8_puzzle state, Expense_8_puzzle goal) {
        int cost = 0;
        for (int i = 0; i < 3; i++) {
            for (int j = 0; j < 3; j++) {
                int tile = state.mat[i][j];
                if (tile != 0) {
                    int goalI = (tile - 1) / 3;
                    int goalJ = (tile - 1) % 3;
                    int moves = Math.abs(i - goalI) + Math.abs(j - goalJ);
                    cost += moves * tile;
                }
            }
        }
        return cost;
    }

    // Parent chain from current state to root (no reversal)
    static List<Expense_8_puzzle> getParentChain(Expense_8_puzzle state) {
        List<Expense_8_puzzle> chain = new ArrayList<>();
        while (state != null) {
            chain.add(state);  // Start from current state
            state = state.predecessor;
        }
        return chain;
    }

    // Log state and its predecessor chain
    static void dumpLogger(Expense_8_puzzle presentPuzzleMat, List<Expense_8_puzzle> successors, Set<String> closedSet, Collection<Expense_8_puzzle> frontier, int NPop, String logFile, Expense_8_puzzle goalState, String method) throws IOException {
        if (logFile != null) {
            BufferedWriter fileEditor = new BufferedWriter(new FileWriter(logFile, true));

            if (method.equals("a*") || method.equals("greedy")) {
                fileEditor.write(String.format("Generating successors to < state = %s, action = {%s} g(n) = %d, d = %d, f(n) = %d, Parent = Pointer to %s >:\n",
                        Arrays.deepToString(presentPuzzleMat.mat), presentPuzzleMat.action == null ? "Start" : presentPuzzleMat.action, presentPuzzleMat.cost,
                        presentPuzzleMat.depth, presentPuzzleMat.cost + estimate_heuristicValues(presentPuzzleMat, goalState),
                        presentPuzzleMat.predecessor == null ? "None" : Arrays.deepToString(presentPuzzleMat.predecessor.mat)));
            } else {
                fileEditor.write(String.format("Generating successors to < state = %s, action = {%s} cost = %d, d = %d, Parent = Pointer to %s >:\n",
                        Arrays.deepToString(presentPuzzleMat.mat), presentPuzzleMat.action == null ? "Start" : presentPuzzleMat.action, presentPuzzleMat.cost,
                        presentPuzzleMat.depth, presentPuzzleMat.predecessor == null ? "None" : Arrays.deepToString(presentPuzzleMat.predecessor.mat)));
            }

            fileEditor.write("\tsuccessors generated -> \n");
            fileEditor.write("\tClosed: " + closedSet + "\n");
            fileEditor.write("\tFringe: [\n");

            // Iterate over frontier and log the state and its predecessor chain
            for (Expense_8_puzzle state : frontier) {
                List<Expense_8_puzzle> predecessorChain = getParentChain(state);  // Correct predecessor chain

                // Print chain from current state to root
                StringBuilder predecessorChainStr = new StringBuilder();
                for (int i = 0; i < predecessorChain.size(); i++) {  // Keep order as-is
                    Expense_8_puzzle p = predecessorChain.get(i);
                    predecessorChainStr.append(String.format("< state = %s, action = {Move %s}, g(n) = %d, d = %d, f(n) = %d >",
                            Arrays.deepToString(p.mat), p.action, p.cost, p.depth, method.equals("a*") || method.equals("greedy") ? (p.cost + estimate_heuristicValues(p, goalState)) : p.cost));
                    if (i != predecessorChain.size() - 1) {  // Append arrow unless last state
                        predecessorChainStr.append(" -> ");
                    }
                }
                fileEditor.write("\t\t" + predecessorChainStr.toString() + "\n");
            }

            fileEditor.write("\t]\n");
            fileEditor.close();
        }
    }


    // DLS (Depth-Limited Search)
    static Expense_8_puzzle dls(Expense_8_puzzle start, Expense_8_puzzle goal, int DL, boolean logBoolean, String logFile) throws IOException {
        Stack<Expense_8_puzzle> frontier = new Stack<>();
        Set<String> closedSet = new HashSet<>();
        frontier.push(start);
        int NPop = 0, NGen = 1, NExp = 0, MFSize = 0;

        while (!frontier.isEmpty()) {
            MFSize = Math.max(MFSize, frontier.size());
            Expense_8_puzzle presentPuzzleMat = frontier.pop();
            NPop++;

            if (presentPuzzleMat.depth > DL) {
                continue;
            }

            if (Arrays.deepEquals(presentPuzzleMat.mat, goal.mat)) {
                System.out.println("Nodes Expanded: " + NExp);
                System.out.println("Nodes Generated: " + NGen);
                System.out.println("Nodes Popped: " + NPop);
                System.out.println("Max Fringe Size: " + MFSize);
                return presentPuzzleMat;
            }

            String matStr = Arrays.deepToString(presentPuzzleMat.mat);
            if (closedSet.contains(matStr)) {
                continue;
            }

            closedSet.add(matStr);
            NExp++;
            List<Expense_8_puzzle> successors = presentPuzzleMat.getExpansions();
            for (Expense_8_puzzle neighbor : successors) {
                if (neighbor.depth <= DL) {
                    frontier.push(neighbor);
                    NGen++;
                }
            }

            if (logBoolean) {
                dumpLogger(presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "dls");
            }
        }
        return null;
    }


    // BFS
    static Expense_8_puzzle bfs(Expense_8_puzzle start, Expense_8_puzzle goal, boolean logBoolean, String logFile) throws IOException {
        Queue<Expense_8_puzzle> frontier = new LinkedList<>();
        Set<String> closedSet = new HashSet<>();
        frontier.add(start);
        int NPop = 0, NGen = 1, NExp = 0, MFSize = 0;

        while (!frontier.isEmpty()) {
            MFSize = Math.max(MFSize, frontier.size());
            Expense_8_puzzle presentPuzzleMat = frontier.poll();
            NPop++;

            if (Arrays.deepEquals(presentPuzzleMat.mat, goal.mat)) {
                System.out.println("Nodes Expanded: " + NExp);
                System.out.println("Nodes Generated: " + NGen);
                System.out.println("Nodes Popped: " + NPop);
                System.out.println("Max Fringe Size: " + MFSize);
                return presentPuzzleMat;
            }

            String matStr = Arrays.deepToString(presentPuzzleMat.mat);
            if (closedSet.contains(matStr)) {
                continue;
            }

            closedSet.add(matStr);
            NExp++;
            List<Expense_8_puzzle> successors = presentPuzzleMat.getExpansions();
            for (Expense_8_puzzle neighbor : successors) {
                frontier.add(neighbor);
                NGen++;
            }

            if (logBoolean) {
                dumpLogger(presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "bfs");
            }
        }
        return null;
    }



    // IDS
    static Expense_8_puzzle ids(Expense_8_puzzle start, Expense_8_puzzle goal, boolean logBoolean, String logFile) throws IOException {
        int DL = 0;
        while (true) {
            Expense_8_puzzle result = dls(start, goal, DL, logBoolean, logFile);
            if (result != null) {
                return result;
            }
            DL++;
        }
    }

    // UCS
    static Expense_8_puzzle ucs(Expense_8_puzzle start, Expense_8_puzzle goal, boolean logBoolean, String logFile) throws IOException {
        PriorityQueue<Expense_8_puzzle> frontier = new PriorityQueue<>();
        Set<String> closedSet = new HashSet<>();
        frontier.add(start);
        int NPop = 0, NGen = 1, NExp = 0, MFSize = 0;

        while (!frontier.isEmpty()) {
            MFSize = Math.max(MFSize, frontier.size());
            Expense_8_puzzle presentPuzzleMat = frontier.poll();
            NPop++;

            if (Arrays.deepEquals(presentPuzzleMat.mat, goal.mat)) {
                System.out.println("Nodes Expanded: " + NExp);
                System.out.println("Nodes Generated: " + NGen);
                System.out.println("Nodes Popped: " + NPop);
                System.out.println("Max Fringe Size: " + MFSize);
                return presentPuzzleMat;
            }

            String matStr = Arrays.deepToString(presentPuzzleMat.mat);
            if (closedSet.contains(matStr)) {
                continue;
            }

            closedSet.add(matStr);
            NExp++;
            List<Expense_8_puzzle> successors = presentPuzzleMat.getExpansions();
            for (Expense_8_puzzle neighbor : successors) {
                frontier.add(neighbor);
                NGen++;
            }

            if (logBoolean) {
                dumpLogger(presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "ucs");
            }
        }
        return null;
    }

    // Greedy
    static Expense_8_puzzle greedy(Expense_8_puzzle start, Expense_8_puzzle goal, boolean logBoolean, String logFile) throws IOException {
        PriorityQueue<Expense_8_puzzle> frontier = new PriorityQueue<>(Comparator.comparingInt(s -> estimate_heuristicValues(s, goal)));
        Set<String> closedSet = new HashSet<>();
        frontier.add(start);
        int NPop = 0, NGen = 1, NExp = 0, MFSize = 0;

        while (!frontier.isEmpty()) {
            MFSize = Math.max(MFSize, frontier.size());
            Expense_8_puzzle presentPuzzleMat = frontier.poll();
            NPop++;

            if (Arrays.deepEquals(presentPuzzleMat.mat, goal.mat)) {
                System.out.println("Nodes Expanded: " + NExp);
                System.out.println("Nodes Generated: " + NGen);
                System.out.println("Nodes Popped: " + NPop);
                System.out.println("Max Fringe Size: " + MFSize);
                return presentPuzzleMat;
            }

            String matStr = Arrays.deepToString(presentPuzzleMat.mat);
            if (closedSet.contains(matStr)) {
                continue;
            }

            closedSet.add(matStr);
            NExp++;
            List<Expense_8_puzzle> successors = presentPuzzleMat.getExpansions();
            for (Expense_8_puzzle neighbor : successors) {
                frontier.add(neighbor);
                NGen++;
            }

            if (logBoolean) {
                dumpLogger(presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "greedy");
            }
        }
        return null;
    }

    // A*
    static Expense_8_puzzle aStar(Expense_8_puzzle start, Expense_8_puzzle goal, boolean logBoolean, String logFile) throws IOException {
        PriorityQueue<Expense_8_puzzle> frontier = new PriorityQueue<>(Comparator.comparingInt(s -> s.cost + estimate_heuristicValues(s, goal)));
        Set<String> closedSet = new HashSet<>();
        frontier.add(start);
        int NPop = 0, NGen = 1, NExp = 0, MFSize = 0;

        while (!frontier.isEmpty()) {
            MFSize = Math.max(MFSize, frontier.size());
            Expense_8_puzzle presentPuzzleMat = frontier.poll();
            NPop++;

            if (Arrays.deepEquals(presentPuzzleMat.mat, goal.mat)) {
                System.out.println("Nodes Expanded: " + NExp);
                System.out.println("Nodes Generated: " + NGen);
                System.out.println("Nodes Popped: " + NPop);
                System.out.println("Max Fringe Size: " + MFSize);
                return presentPuzzleMat;
            }

            String matStr = Arrays.deepToString(presentPuzzleMat.mat);
            if (closedSet.contains(matStr)) {
                continue;
            }

            closedSet.add(matStr);
            NExp++;
            List<Expense_8_puzzle> successors = presentPuzzleMat.getExpansions();
            for (Expense_8_puzzle neighbor : successors) {
                frontier.add(neighbor);
                NGen++;
            }

            if (logBoolean) {
                dumpLogger(presentPuzzleMat, successors, closedSet, frontier, NPop, logFile, goal, "a*");
            }
        }
        
        return null;
    }

    public static void main(String[] args) throws IOException {
        if (args.length < 2) {
            System.out.println("Usage: java Expense_8_puzzle <start-file> <goal-file> <method> <log-file>");
            return;
        }

        String startFile = args[0];
        String goalFile = args[1];
       
        String method = "a*";
        boolean logBoolean = false;
        String logFile = "traceDateTime.txt" ;
        // trace-Date-Time.txt 
        logFile = "trace-" + new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss").format(new Date()) + ".txt";
        if(args.length > 2){
            if(args[2].equals("true") || args[2].equals("false")){
                logBoolean = Boolean.parseBoolean(args[2]);
            }
            else{
                method = args[2];
            }
        }
        if(args.length > 3){
            logBoolean = Boolean.parseBoolean(args[3]);
        }


        Expense_8_puzzle startState = Expense_8_puzzle.convertToMatFromFile(startFile);
        Expense_8_puzzle goalState = Expense_8_puzzle.convertToMatFromFile(goalFile);

        Expense_8_puzzle answer = null;

        switch (method) {
            case "bfs":
                answer = Expense_8_puzzle.bfs(startState, goalState, logBoolean, logFile);
                break;
            case "dfs":
                answer = Expense_8_puzzle.dls(startState, goalState, Integer.MAX_VALUE, logBoolean, logFile);
                break;
            case "dls":
                int DL = 10;
                Scanner sc = new Scanner(System.in);
                System.out.println("Enter the depth limit: ");
                String inputTaken = sc.nextLine();
                if(inputTaken.length() > 0 && inputTaken.matches("[0-9]+")){
                    DL = Integer.parseInt(inputTaken);
                }
                else{
                    System.out.println("Invalid input, using default depth limit of 10");
                }

                answer = Expense_8_puzzle.dls(startState, goalState, DL, logBoolean, logFile);
                break;
            case "ids":
                answer = Expense_8_puzzle.ids(startState, goalState, logBoolean, logFile);
                break;
            case "ucs":
                answer = Expense_8_puzzle.ucs(startState, goalState, logBoolean, logFile);
                break;
            case "greedy":
                answer = Expense_8_puzzle.greedy(startState, goalState, logBoolean, logFile);
                break;
            case "a*":
                answer = Expense_8_puzzle.aStar(startState, goalState, logBoolean, logFile);
                break;
            default:
                System.out.println("Unknown method!");
                return;
        }

        if (answer != null) {
            System.out.println("Solution Found at depth " + answer.depth + " with cost of " + answer.cost);
            List<String> steps = answer.parentChainGenerator();
            for (String step : steps) {
                System.out.println("\t" + step);
            }
        } else {
            System.out.println("No answer found.");
        }
    }
}
