17

I am trying to take a base64 encoded string and return it as an image in php using $_POST. On line one if I use $_POST['imgdata'] it returns error from the preg_match if i were hard code the base64 string instead of using $_POST it all works and returns the image. how can i make this work by using the $_POST

works

$imgstr = '';

does not work

$imgstr = $_POST['imgdata'];

full code

$imgstr = $_POST['imgdata'];   
// Grab the MIME type and the data with a regex for convenience
if (!preg_match('/data:([^;]*);base64,(.*)/', $imgstr, $matches)) {
    die("error");
}

// Decode the data
$content = base64_decode($matches[2]);

// Output the correct HTTP headers (may add more if you require them)
header('Content-Type: '.$matches[1]);
header('Content-Length: '.strlen($content));

// Output the actual image data
echo $content;
Daniel
  • 4,202
  • 11
  • 50
  • 68
  • The POSTed value looks like...? – Ignacio Vazquez-Abrams Jul 20 '12 at 00:26
  • @Ignacio Vazquez-Abrams i added the full string – Daniel Jul 20 '12 at 00:32
  • have you tried to echo $_POST['imgdata'] and compare? – haynar Jul 20 '12 at 00:37
  • @haynar if i echo $_POST it returns blank page and var_dump returns null – Daniel Jul 20 '12 at 00:41
  • so the problem is at client side, when sending the data, can you show us the client side code? – haynar Jul 20 '12 at 00:44
  • @haynar i havent created any client side code yet i'm doing it from the browser url like this `http://mywebsite.com/dataurl_decode.php?imgdata=‌​goAAAANSUhEUgAAAMgAAAAoCAYAAAC7HLUcAAADr0lEQVR4Xu2brZIqMRCFsy8A` – Daniel Jul 20 '12 at 00:46
  • 3
    ah that's why it is null, you are sending the data as GET, but trying to get it as POST, use $_GET instead – haynar Jul 20 '12 at 00:47
  • @haynar i tried both get and post and still get null. by the way i get error "Notice: Undefined index: imgdata" – Daniel Jul 20 '12 at 00:50
  • it is better to implement the client-side scripts, because the data which you can send via GET is limited and in case of base64-encoded images I guess the length of data will exceed that limit, so at first make a simple implementation at client-side with a POST data – haynar Jul 20 '12 at 00:51
  • 1
    Might want to try `$_REQUEST`. – Cole Tobin Jul 20 '12 at 03:16

7 Answers7

17

As was noted the "+" should not be missed, the rest is straight forward. Use $_REQUEST if you are not sure is it post or get.

// requires php5
define('UPLOAD_DIR', 'images/');
$img = $_REQUEST['img'];
$img = str_replace('data:image/png;base64,', '', $img);
$img = str_replace(' ', '+', $img);
$data = base64_decode($img);
$file = UPLOAD_DIR . uniqid() . '.png';
$success = file_put_contents($file, $data);
print $success ? $file : 'Unable to save the file.';
Logic Wreck
  • 366
  • 1
  • 9
  • Don't you also have to replace that + signs? In iOS that would be encodedString = [encodedString stringByReplacingOccurrencesOfString:@"+" withString:@"%2B"]; – coolcool1994 Aug 11 '13 at 03:49
7

This is the One., Would you like the following i think.,

<?php
$imgstr = 'image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAMgAAAAoCAYAAAC7HLUcAAADtUlEQVR4Xu2aLZYqMRCFMxuAFaARg8WwAnaABQEWgwUBFoMFARaDRqAxWBBoVgAbmPcu7/Q7PX3SP0k16TTcnDNqOpXKrXykKsmXUurn7x+bvQJf9l3Z03cFEFwCIosSAZHp53VvAiIPDwGRa+itBQIiDw0BkWvorQUCIg8NAZFr6K0FAiIPDQGRa+itBQIiDw0BkWvorQUCIg8NAZFr6K0FAiIPDQGRa+itBTEg7XZb1Wo11Wg0VKVS+T/Rx+Ohzuezut1uar/feyVAp9NR8Dvaer2ejZ8ExEa1kvSxBgQLbDQaPeFIa4BkPp97AQr8XiwWWpfr9XraVHT/JyA2qpWkjxUg+AWeTqfGU5xMJmq73Rr3y6vD9/e32mw2v3a6sG0CkpfS72PHGBAsst1uZ60A0pjj8Wjd37Yj0j/AAf/jGgGxVfd9+xkDgp0DO0i0oeZYLpfqcrk8F+FgMND+UqMeGQ6HzhVdr9eq1WoljktAnIfF+wGNATmdTtqFH02fsBixKHUgNZtNp8LEQR11goA4DUspBjMCJC69wu6hW/TX6zXPYthKUF29BH/DJ26BYQJiJfFbdzICJFAinKoAGiy4aPGN063D4VDoDqIDGr7iRE13yEBA3nqtW03OCpAsI+EoVXfX4KoGAaA4TIjezXS7XVWtVrXpHwHJEtnP+uYlgCTl/C5OseJOrII6Ka4+IiCftfizzDZXQLAwcXmoO+WCM652D93uhbRqtVo9NSEgWZYGv4ECuQGClAYLM+6eAce/SG9QA7yy9fv9J6ThhvoIu0e4htKdsHEHeWVkymk7F0BQa8xms9gbalwM4u7j1XDoTqx0YHIHKediLcJrMSC6X+zwRJDWIL1x0XR3NBj/fr//Gh67nS4NDPzUncol+M+3WC6CW9AYIkCSinEssvF47PSBYty9i6m22PEMXvYSEFOBS/S9NSBJcLhKqaI6E5ASrbySuGoFSFJaFS2IXepAQFyq/RljGQOS9po3y0td5PoonvNuBCRvRWnPGJAsr2LTZH3VZWHaa93AL0AePQrG/4K6A0W9AcCsQdICXuL/GwES977KdP6vAiSrHzzmzaoUvzMCJO1IN6ucBCSrUvyuaAWMAIl7gGg6CQJiqhi/L0oBI0CKctLzcVmDeB4giXsERKLev74ERK6htxYIiDw0BESuobcWCIg8NARErqG3FgiIPDQERK6htxYIiDw0BESuobcWCIg8NARErqG3FgiIPDQERK6htxYIiDw0BESuobcW/gDZOWY4lzJl1QAAAABJRU5ErkJggg==';
$new_data=explode(";",$imgstr);
$type=$new_data[0];
$data=explode(",",$new_data[1]);
header("Content-type:".$type);
echo base64_decode($data[1]);
?>

The above code will generates the output as per your wish.,Try this..This may help you.

Sorry For The Delay Reply.,Now Only I Recover My StackOverFlow Account After a long time.

Note : Ignore the empty space's in before the starting '' tag on the PHP file you are using for image decode.,If any empty is on that file means it throws Error message and it doesn't create or show the original image after the decodes.

John Peter
  • 2,870
  • 3
  • 27
  • 46
  • i am trying to use the $_POST variable to send the string to a php page – Daniel Aug 14 '12 at 18:34
  • Just add the hidden field and place the value to get in POST Variable.,like and get the value in post while form submitting $_POST['imgstr'] also assign it to $imgstr=$_POST['imgstr']; – John Peter Aug 15 '12 at 09:26
3
<?php
$data = 'iVBORw0KGgoAAAANSUhEUgAAABwAAAASCAMAAAB/2U7WAAAABl'
       . 'BMVEUAAAD///+l2Z/dAAAASUlEQVR4XqWQUQoAIAxC2/0vXZDr'
       . 'EX4IJTRkb7lobNUStXsB0jIXIAMSsQnWlsV+wULF4Avk9fLq2r'
       . '8a5HSE35Q3eO2XP1A1wQkZSgETvDtKdQAAAABJRU5ErkJggg==';
$data = base64_decode($data);

$im = imagecreatefromstring($data);
if ($im !== false) {
    header('Content-Type: image/png');
    imagepng($im);
    imagedestroy($im);
}
else {
    echo 'An error occurred.';
}
?> 

source: documentation.

check out imagecreatefromstring.

Prasanth
  • 5,230
  • 2
  • 29
  • 61
2

Not sure why the regex isn't working for you, I copied the base64 post data and your code and it worked fine. You can try this instead which doesn't use regex and may be a little faster and use less memory.

$imgstr = $_GET['imgdata'];

list($type, $imgstr) = explode(';', $imgstr);
list(, $type)        = explode(':', $type);
list(, $imgstr)      = explode(',', $imgstr);
$content = base64_decode($imgstr);
drew010
  • 68,777
  • 11
  • 134
  • 162
  • it doesnt work. if i try to put this in my browser `http://mywebsite.com/dataurl_decode.php?imgdata=` just returns a blank page – Daniel Jul 20 '12 at 00:41
  • A blank page may indicate a parse error and you have `display_errors` set to off. By the way, that is `$_GET` data, not `$_POST`, so that's probably why the regex isn't working for you. – drew010 Jul 20 '12 at 00:44
  • Probably because `$_POST['imgdata']` is not set. If you are passing the data in the URL query string, you need to use `$_GET['imgdata']` instead. – drew010 Jul 20 '12 at 00:50
  • 3
    Also the image is breaking because the `+` in the base64 data is getting converted to a `' '` because URLs treat the `+` as a space. You shouldn't pass that via get, instead use post or call `urlencode` on the data before you put it in the query string. – drew010 Jul 20 '12 at 00:55
  • Or just do a `str_replace(' ', '+', ...);` – Cole Tobin Jul 20 '12 at 03:17
1

Please check below code it is working fine for me,

$data = '';
$data = str_replace('data:image/png;base64,', '', $data);
$data = str_replace(' ', '+', $data);
$data = base64_decode($data); // Decode image using base64_decode
$file = uniqid() . '.png'; //Now you can put this image data to your desired file using file_put_contents function like below:
$success = file_put_contents($file, $data);
HEMAL
  • 420
  • 3
  • 14
1

You can capture parts by this regex :

^data:([a-zA-Z]+/[a-zA-Z]+);base64\,([a-zA-Z0-9+\=/]+)$

0
$imgstr = '';
preg_match("/data\:image\/(.*)\;base64/",$imgstr, $match);
echo $match[1];

^^^^ Return file extension.

$data = str_replace('data:image/png;base64,', '', $imgstr);
data = str_replace(' ', '+', $data);
$data = base64_decode($data);
file_put_contents("/files/youfilemane.".$match[1],$data);

^^^^ Create file from base64.

Fiodorov Andrei
  • 1,778
  • 1
  • 11
  • 26
  • 1
    Colons and semicolons do not require escaping in a regex pattern. The pattern to extract the extension can be as simple as `~data:image/\K[^;]+~` then just use the `[0]` match – mickmackusa Dec 02 '19 at 23:35