{
  "id": "astropy__astropy-8707",
  "question": "Header.fromstring does not accept Python 3 bytes\nAccording to [the docs](http://docs.astropy.org/en/stable/_modules/astropy/io/fits/header.html#Header.fromstring), the method `Header.fromstring` \"...creates an HDU header from a byte string containing the entire header data.\"\r\n\r\nBy \"byte string\" here it really means the `str` type which on Python 2 could be raw binary data, but on Python 3 explicitly is not.   In fact it does work on Python 3's unicode `str`s, but here it assumes that the data can be ASCII-encoded.\r\n\r\nIts counterpart, `Header.fromfile` will work with files opened in text or binary mode.  So probably the simplest solution for now (as opposed to adding new methods or something like that) is to change `Header.fromstring` to accept unicode or bytes string types.\r\n\r\n`Card.fromstring` likely needs a similar treatment.\n",
  "answer": "{\"task_id\": \"8c009d2e-1302-4fb0-afc2-e0db0a88b93c\", \"workflow_run_id\": \"89b8b857-d0b5-40a8-a670-5caf739a964f\", \"data\": {\"id\": \"89b8b857-d0b5-40a8-a670-5caf739a964f\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,6 +210,8 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\", \"--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,6 +210,11 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n+        # Handle bytes input by decoding to string\\n+        if isinstance(data, bytes):\\n+            try:\\n+                data = data.decode('ascii')\\n+            except UnicodeDecodeError:\\n+                raise ValueError(\\\"Header data contains non-ASCII bytes\\\")\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\", \"--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,8 +210,15 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n-        Creates an HDU header from a byte string containing the entire\\n-        header data.\\n+        Creates an HDU header from a string or bytes containing the entire\\n+        header data.\\n+        \\n+        Parameters\\n+        ----------\\n+        data : str or bytes\\n+            The header data as a string or bytes object\\n+        \\\"\\\"\\\"\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\", \"--- a/astropy/io/fits/card.py\\n+++ b/astropy/io/fits/card.py\\n@@ -150,6 +150,8 @@ class Card:\\n     @classmethod\\n     def fromstring(cls, image):\\n         \\\"\\\"\\\"\\n+        if isinstance(image, bytes):\\n+            image = image.decode('ascii')\\n         return cls._from_string(image)\\n     \\n     @classmethod\", \"--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,6 +210,9 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\\n--- a/astropy/io/fits/card.py\\n+++ b/astropy/io/fits/card.py\\n@@ -150,6 +150,9 @@ class Card:\\n     @classmethod\\n     def fromstring(cls, image):\\n         \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(image, bytes):\\n+            image = image.decode('ascii')\\n         return cls._from_string(image)\", \"diff\\n--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,8 +210,15 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n-        Creates an HDU header from a byte string containing the entire\\n-        header data.\\n+        Creates an HDU header from a string or bytes containing the entire\\n+        header data.\\n+        \\n+        Parameters\\n+        ----------\\n+        data : str or bytes\\n+            The header data as a string or bytes object\\n+        \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\\n--- a/astropy/io/fits/card.py\\n+++ b/astropy/io/fits/card.py\\n@@ -150,6 +150,9 @@ class Card:\\n     @classmethod\\n     def fromstring(cls, image):\\n         \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(image, bytes):\\n+            image = image.decode('ascii')\\n         return cls._from_string(image)\\n\", \"diff\\n--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,8 +210,15 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n-        Creates an HDU header from a byte string containing the entire\\n-        header data.\\n+        Creates an HDU header from a string or bytes containing the entire\\n+        header data.\\n+        \\n+        Parameters\\n+        ----------\\n+        data : str or bytes\\n+            The header data as a string or bytes object\\n+        \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\\n--- a/astropy/io/fits/card.py\\n+++ b/astropy/io/fits/card.py\\n@@ -150,6 +150,9 @@ class Card:\\n     @classmethod\\n     def fromstring(cls, image):\\n         \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(image, bytes):\\n+            image = image.decode('ascii')\\n         return cls._from_string(image)\\n\", \"diff\\n--- a/astropy/io/fits/header.py\\n+++ b/astropy/io/fits/header.py\\n@@ -210,8 +210,15 @@ class Header:\\n     @classmethod\\n     def fromstring(cls, data, sep=''):\\n         \\\"\\\"\\\"\\n-        Creates an HDU header from a byte string containing the entire\\n-        header data.\\n+        Creates an HDU header from a string or bytes containing the entire\\n+        header data.\\n+        \\n+        Parameters\\n+        ----------\\n+        data : str or bytes\\n+            The header data as a string or bytes object\\n+        \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(data, bytes):\\n+            data = data.decode('ascii')\\n         cards = []\\n         for line in data.split(sep or '\\\\n'):\\n             if line:\\n--- a/astropy/io/fits/card.py\\n+++ b/astropy/io/fits/card.py\\n@@ -150,6 +150,9 @@ class Card:\\n     @classmethod\\n     def fromstring(cls, image):\\n         \\\"\\\"\\\"\\n+        # Convert bytes to string if necessary\\n+        if isinstance(image, bytes):\\n+            image = image.decode('ascii')\\n         return cls._from_string(image)\\n\"]}, \"error\": \"\", \"elapsed_time\": 94.120776, \"total_tokens\": 35008, \"total_steps\": 30, \"created_at\": 1754646260, \"finished_at\": 1754646354}}"
}