Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fmt: cannot read more than 254 characters in Windows when calling Fscanln #42551

Open
jimen0 opened this issue Nov 12, 2020 · 7 comments
Open
Labels
NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one. OS-Windows
Milestone

Comments

@jimen0
Copy link
Contributor

jimen0 commented Nov 12, 2020

What version of Go are you using (go version)?

$ go version
go version go1.15.4 darwin/amd64
$ gotip version
go version devel +b34b0aaf69 Thu Nov 12 15:28:43 2020 +0000 darwin/amd64

Does this issue reproduce with the latest release?

Yes. It also reproduces with current tip go version devel +b34b0aaf69 Thu Nov 12 15:28:43 2020 +0000 darwin/amd64

What operating system and processor architecture are you using (go env)?

go env Output
$ go env
N/A

What did you do?

package main

import (
	"fmt"
	"os"
)

func main() {
	var s string
	_, err := fmt.Fscanln(os.Stdin, &s)
	if err != nil {
		panic(err)
	}

	fmt.Println(s)
}
$ GOOS=windows go build -o reproducer_stdin.exe main.go
$ GOOS=windows gotip build -o reproducer_stdin_tip.exe main.go

What did you expect to see?

On the Windows host, I type A character 257 times after running the program, then press enter key. The Go program reads them correctly.

PS C:\Users\mig\Downloads> .\reproducer_stdin_tip.exe | Measure-Object -char | Select-Object -expand Characters
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
257
PS C:\Users\mig\Downloads> .\reproducer_stdin.exe | Measure-Object -char | Select-Object -expand Characters
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
257

What did you see instead?

On the Windows host, I typed A character 257 times after running the program, then presed enter key. The Go program stdin froze after 254 characters.

PS C:\Users\mig\Downloads> .\reproducer_stdin_tip.exe | Measure-Object -char | Select-Object -expand Characters
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
254
PS C:\Users\mig\Downloads> .\reproducer_stdin.exe | Measure-Object -char | Select-Object -expand Characters
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
254

NOTE: this seems to also be borken when copying&pasting text.

Note: this was tested using a Windows (Windows Server 2012 R2 Datacenter) Azure machine with default settings and environment. Just downloaded the cross-compiled binaries into it and ran them.
Edit: This is also reproducible using Windows Command Prompt using an Azure Windows (Windows 10 Pro) default machine.


I don't really know if this is related to Go at all or an issue with Powershell itself, but I thought it was worth submitting an issue. This was initially discovered by @karelorigin.

Best regards,
Miguel

@jimen0 jimen0 changed the title fmt/Fscanln: cannot read more than 254 characters in Windows Powershell fmt/Fscanln: cannot read more than 254 characters in Windows Nov 12, 2020
@cagedmantis cagedmantis changed the title fmt/Fscanln: cannot read more than 254 characters in Windows fmt: cannot read more than 254 characters in Windows when calling Fscanln Nov 13, 2020
@cagedmantis cagedmantis added OS-Windows NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one. labels Nov 13, 2020
@cagedmantis cagedmantis added this to the Backlog milestone Nov 16, 2020
@networkimprov
Copy link

cc @alexbrainman @mattn

@reznik99
Copy link

reznik99 commented Jan 4, 2022

Has it been confirmed if it's a Go issue?
I have tried CMD and Powershell still having this issue.

Input is truncated to a maximum of 254 characters whether typing or pasting.
Works fine on Linux build

@sgtxn
Copy link

sgtxn commented Feb 4, 2023

Same, stumbled upon this today when trying to paste a long authorization code into fmt.Scan input, seems to occur no matter if I'm using the old cmd.exe, the new Windows Terminal or the VSCode built-in terminal. Also occurs in cmd, Powershell and Git Bash.

Wrapping os.Stdin in bufio.NewReader solves the issue though:

var code string
in := bufio.NewReader(os.Stdin)
if _, err := fmt.Fscan(in, &code); err != nil {
    log.Fatal(err)
}

@karelorigin
Copy link

@reznik99, verifying that it is Go specific is actually quite simple. Try the following python script:

s = input("input more than 254 characters:\n")
print("Received: " + s)

You can execute this in your browser devtools to quickly generate a long string:

copy("a".repeat(254) + "b")
> python3 input.py
input more than 254 characters:
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaab
Received: aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaab

@vault-thirteen
Copy link

This bug still exists in the latest stable version of Go language (1.22.9).

@jimen0
Copy link
Contributor Author

jimen0 commented Nov 21, 2024

Back when @karelorigin and I raised this I continued my research. The problem doesn't seem to be Go itself but the Windows console that stops reading input from the user after 254 bytes (256 if you add the required \r\n).

Not ideal but this seems to work on Windows 10:

package main

import (
	"fmt"

	"golang.org/x/sys/windows"
)

const size = 1024

func main() {
	stdin, err := windows.GetStdHandle(windows.STD_INPUT_HANDLE)
	if err != nil {
		panic(err)
	}

	buf := make([]uint16, size)
	var read uint32
	err = windows.ReadConsole(stdin, &buf[0], size, &read, nil)
	if err != nil {
		panic(err)
	}

	output := windows.UTF16ToString(buf[:read])
	fmt.Print(output)
	fmt.Printf("\nInput length: %d bytes (utf8 string)\n", len(output))
}

Copy a large string such as this one and try it:

$ python3 -c "print('A'*254 + 'B'*2 + 'C'*200)" | pbcopy

# AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABBCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
PS C:\Users\User\Downloads> .\reproducer_stdin.exe
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABBCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABBCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC

Input length: 458 bytes (utf8 string)

That is 456 utf8 characters plus \r\n.

The ReadConsole implementation is at https://cs.opensource.google/go/x/sys/+/refs/tags/v0.27.0:windows/zsyscall_windows.go;l=2995-3000

Hope it helps @vault-thirteen 😄

@vault-thirteen
Copy link

@jimen0 , great job ! This is all cool, but ...

... all this should be done by the developers of Go language. Go's creators should incorporate all this into the standard library. End users of the language should not invent their own language upon the existing language. If people need to invent a second language upon the existing one, then something in the first language is wrong.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one. OS-Windows
Projects
None yet
Development

No branches or pull requests

7 participants