[Python] [Stage #QP2] Autocompletion works locally but fails tests

import sys, shutil, subprocess, os, shlex, msvcrt

builtin_commands = {}

def command(func):
    builtin_commands[func.__name__.split("_")[1]] = func
    return func

@command
def shell_exit(args):
    exit(int(args[0]))

@command
def shell_echo(args):
    sys.stdout.write(" ".join(args) + "\n")

@command
def shell_type(args):
    if args[0] in builtin_commands:
        sys.stdout.write(f"{args[0]} is a shell builtin\n")
    elif path := shutil.which(args[0]):
        sys.stdout.write(f"{args[0]} is {path}\n")
    else:
        sys.stderr.write(f"{args[0]}: not found\n")

@command
def shell_pwd(args):
    sys.stdout.write(f"{os.getcwd()}\n")

@command
def shell_cd(args):
    path = args[0]
    if os.path.isdir(path) or path == "~":
        os.chdir(os.path.expanduser(path))
    else:
        sys.stderr.write(f"cd: {path}: No such file or directory\n")

def get_completion(input):
    return list(filter(lambda cmnd: cmnd.startswith(input), builtin_commands))

def write_and_flush(data):
    sys.stdout.write(data)
    sys.stdout.flush()

def get_input():
    user_input = ""
    print("$ ", end="", flush=True)
    
    while True:
        char = msvcrt.getwch()
        if char == "\r":
            print()
            return user_input
        elif char == "\b":
            if len(user_input) > 0:
                user_input = user_input[:-1]
                print("\b \b", end="", flush=True)
        elif char == "\t":  
            completions = get_completion(user_input.strip())
            if completions:
                completion = completions[0]
                user_input = completion + " "
                
                sys.stdout.write(f"\r$ {user_input}")
                sys.stdout.flush()
        else:  
            user_input += char
            print(char, end="", flush=True)
    
    sys.stdout.write("\n")
    return user_input

def redirect(args, command, operator):
    if len(args) < 3:
        return

    redirect_index = args.index(operator)
    cmd_args = args[:redirect_index]
    file_name = args[redirect_index + 1]

    method = "w" if operator in [">", "1>", "2>"] else "a"

    if command in builtin_commands:
        try:
            with open(file_name, method) as f:
                old_output = sys.stdout if operator in [">", "1>",">>","1>>"] else sys.stderr
                if operator in [">", "1>",">>","1>>"]:
                    sys.stdout = f
                elif operator in ["2>", "2>>"]:
                    sys.stderr = f

                builtin_commands[command](cmd_args)

        except Exception as e:
            sys.stdout.write(f"Error executing command {command}: {e}\n")
        finally:
            if operator in [">", "1>",">>","1>>"]:
                sys.stdout = old_output
            elif operator in ["2>", "2>>"]:
                sys.stderr = old_output
    else:
        executable = shutil.which(command)
        if executable:
            with open(file_name, method) as f:
                if operator in [">", "1>",">>","1>>"]:
                    subprocess.run([command] + cmd_args, stdout=f)
                else:
                    subprocess.run([command] + cmd_args, stderr=f)
        else:
            sys.stdout.write(f"{command}: command not found\n")



def main():
    while True:
        user_input = get_input()
        cmd, *args = shlex.split(user_input)

        redirect_operator = next((operator for operator in [">","1>","2>",">>", "1>>", "2>>"] if operator in args), None)
        

        if redirect_operator:
            redirect(args, cmd, redirect_operator)
        else:       
            if cmd in builtin_commands:
                builtin_commands[cmd](args)
            elif executable := shutil.which(cmd):
                subprocess.run([cmd] + args)
            else:
                sys.stdout.write(f"{cmd}: command not found\n")


if __name__ == "__main__":
    main()

Hey @Jebach0, what errors are you seeing?

@andy1li This is the output I’m getting

remote: [tester::#QP2] Running tests for Stage #QP2 (Autocompletion - Builtin completion)
remote: [tester::#QP2] Running ./your_program.sh
remote: [tester::#QP2] ✓ Received prompt ($ )
remote: [tester::#QP2] Typed “ech”
remote: [tester::#QP2] ✓ Prompt line matches “$ ech”
remote: [tester::#QP2] Pressed “” (expecting autocomplete to “echo”)
remote: [your-program] $ ech
remote: [tester::#QP2] Output does not match expected value.
remote: [tester::#QP2] Expected: "$ echo "
remote: [tester::#QP2] Received: "$ ech "
remote: [tester::#QP2] Assertion failed.
remote: [tester::#QP2] Test failed (try setting ‘debug: true’ in your codecrafters.yml to see more details)
remote:
remote: Try our CLI to run tests faster without Git: How do I install the CLI? - CodeCrafters
remote:
remote: View our article on debugging test failures: How do I debug test failures? - CodeCrafters
remote:
To https://git.codecrafters.io/5ac8d4f248023317

@Jebach0 Nice use of the command decorator! :+1:

The default setup does not allow reading one char at a time from stdin:

Input is currently buffered until enter is pressed. To handle character-by-character input, you may want to research raw mode or use readline.

Let me know if you’d like further clarification!

Closing this thread due to inactivity. If you still need assistance, feel free to reopen or start a new discussion!