IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Langage PHP Discussion :

fgetcsv trompé malgré le délimiteur


Sujet :

Langage PHP

  1. #1
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut fgetcsv trompé malgré le délimiteur
    Bonjour,

    Mon fichier CSV contient :
    "ACTIVATEDATE","DEACTIVATEDATE"
    "Jan 01, 2021",""
    
    Donc le délimiteur est ".
    Pour le lire, j'ai
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    while (($row = CSVParser::getRow()) !== false) {
                $lineCSV = array_combine(CSVParser::getHeader(), explode(",",$row[0]));
    avec
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    public static function getRow(int $buffer = 4096) {
            return(fgetcsv(self::$handle, $buffer, self::$separator, self::$enclosure,""));
        }
    et

    Problème du explode : s'il y a une virgule dans $row[0], il va extraire 2 éléments au lieu d'un seul. Je dois faire un explode car tous les éléments d'une ligne du CSV sont contenus dans la même cellule. Le souci ici est que le header va contenir 2 éléments mais la ligne 3. Comment faut-il faire, SVP ?

  2. #2
    Expert éminent
    Avatar de Séb.
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    5 255
    Détails du profil
    Informations personnelles :
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mars 2005
    Messages : 5 255
    Points : 8 548
    Points
    8 548
    Billets dans le blog
    17
    Par défaut
    Tu ne devrais pas avoir à faire de explode()

    Il faut que $separator vale ,
    et $delimiter "

    Est-ce le cas ?

  3. #3
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Presque ; j'ai vu d'où vient le problème ; j'avais raccourci la ligne par souci de clarté, mais voici une ligne complète :
    SESA1000,"(none)","Software Engineering","SESA84441","Atlassian","JIRA Global Instance","","True","Jan 01, 2021","","","Spain","Miguel","Abad","SCHNEIDER ELECTRIC ESPANA, S.A.U.","FKP8","","Global Finance","BARCELONA (BAC DE RODA)","miquel.abad@se.com","Alistair","Mckelvie","alistair.mckelvie@se.com"
    Tous ces éléments sont regroupés dans une seule cellule du CSV. On voit que tous les éléments sont encadrés par le délimiteur ", sauf le premier !

    Je vais m'absenter environ une heure...

  4. #4
    Expert éminent
    Avatar de Séb.
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    5 255
    Détails du profil
    Informations personnelles :
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mars 2005
    Messages : 5 255
    Points : 8 548
    Points
    8 548
    Billets dans le blog
    17
    Par défaut
    Presque
    Que valent self::$delimiter/$separator dans ton script ?

  5. #5
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Bonsoir, en fait, en voulant réparer, j'ai tout cassé et la dernière sauvegarde date du 30 novembre

    Donc demain, j'essaie de remettre en ordre et je reprendrai cette discussion...Bonne nuit

  6. #6
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Tous comptes faits, y avait pas grand chose à faire, mais y a quand même 2 jours qui se sont écoulés

    On a les propriétés $enclosure='"' et $separator=','. Quand tu parles de $delimiter, je suppose qu'il s'agit en fait de $enclosure...

  7. #7
    Expert éminent
    Avatar de Séb.
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    5 255
    Détails du profil
    Informations personnelles :
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mars 2005
    Messages : 5 255
    Points : 8 548
    Points
    8 548
    Billets dans le blog
    17
    Par défaut
    On a les propriétés $enclosure='"' et $separator=','
    Montre-nous où dans ton code tu obtiens ces valeurs stp.

  8. #8
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
     public static function detectSyntaxFromHeader(): bool
        {
            if (isset(self::$handle)) {
                $row = fgets(self::$handle);
                $row=strtolower($row); //mettre header en minuscules
            } // doit contenir les en-têtes des colonnes}
            if (empty($row)) {
                self::$errors[] = 'The file is empty';
                return false;
            }
     
            // séparateur
            if (mb_stripos($row, ',') !== false) {
                self::$separator = ',';
            } elseif (mb_stripos($row, ';') !== false) {
                self::$separator = ';';
            } else {
                self::$errors[] = 'Unable to detect the CSV field separator';
                return false;
            }
     
            $char = mb_substr($row, 0, 1); // premier caractère
            if ($char === "\xef\xbb\xbf") {  //if first character is BOM, one begins with the next character
                $row  = mb_substr($row, 1);
                $char = mb_substr($row, 0, 1);
            }
     
            if (($char === '"') && (mb_stripos($row, '"'.self::$separator.'"') !== false)) {
                self::$enclosure = '"';
            } elseif (($char === "'") && (mb_stripos($row, "'".self::$separator."'") !== false)) {
                self::$enclosure = "'";
            }
     
            if (self::$enclosure) {
                // on vérifie que le premier séparateur trouvé est bien entouré de lettres ASCII
                $i = mb_stripos($row, self::$separator);
                if ($i) {
                    $fc = mb_substr($row, $i - 1, 1);
                    $lc = mb_substr($row, $i + 1, 1);
                    if (ctype_alpha($fc) && ctype_alpha($lc)) {
                        self::$enclosure = '';
                    }
                }
            }
            else  self::$enclosure = '';
     
            if (self::$enclosure === false) {
                self::$errors[] = 'Unable to detect the CSV text enclosure or this file has no header';
                return false;
            } else {
                // transform header columns into array
                self::$header = explode(self::$separator, str_replace(self::$enclosure, '', trim($row, "\r\n")));
                return true;
            }
        }

  9. #9
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Sinon, j'ai adopté un moyen alambiqué pour m'en sortir :
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
     while (($line_csv = fgetcsv(CSVParser::getHandle(), $buffer, CSVParser::getSeparator(), $enclosure,"")) !== false) {
     
                //on lit une ligne du CSV, on l'analyse et si pas d'erreur, en bdd
     
                 // découper line_csv en un tableau sép="," si besoin
                if (sizeof($line_csv)==1)
                    $line_csv=explode('","', $line_csv[0]);
                //découper 1e cellule avec sép=,
                $first_cell = explode(',',$line_csv[0]);
     
                if (isset($first_cell[1]))
                {
                    // on enlève le 1er elt de line_csv et on met les 2 à la place
                    array_shift($line_csv);
                    array_unshift($line_csv, $first_cell[0], $first_cell[1]);
                }
    y a plus d'explode et je dois traiter aussi les cas où ça se présente bien...

  10. #10
    Expert éminent
    Avatar de Séb.
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    5 255
    Détails du profil
    Informations personnelles :
    Âge : 46
    Localisation : France

    Informations professionnelles :
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mars 2005
    Messages : 5 255
    Points : 8 548
    Points
    8 548
    Billets dans le blog
    17
    Par défaut
    Citation Envoyé par laurentSc Voir le message
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
     public static function detectSyntaxFromHeader(): bool
        {
            if (isset(self::$handle)) {
                $row = fgets(self::$handle);
                $row=strtolower($row); //mettre header en minuscules
            } // doit contenir les en-têtes des colonnes}
            if (empty($row)) {
                self::$errors[] = 'The file is empty';
                return false;
            }
     
            // séparateur
            if (mb_stripos($row, ',') !== false) {
                self::$separator = ',';
            } elseif (mb_stripos($row, ';') !== false) {
                self::$separator = ';';
            } else {
                self::$errors[] = 'Unable to detect the CSV field separator';
                return false;
            }
     
            $char = mb_substr($row, 0, 1); // premier caractère
            if ($char === "\xef\xbb\xbf") {  //if first character is BOM, one begins with the next character
                $row  = mb_substr($row, 1);
                $char = mb_substr($row, 0, 1);
            }
     
            if (($char === '"') && (mb_stripos($row, '"'.self::$separator.'"') !== false)) {
                self::$enclosure = '"';
            } elseif (($char === "'") && (mb_stripos($row, "'".self::$separator."'") !== false)) {
                self::$enclosure = "'";
            }
     
            if (self::$enclosure) {
                // on vérifie que le premier séparateur trouvé est bien entouré de lettres ASCII
                $i = mb_stripos($row, self::$separator);
                if ($i) {
                    $fc = mb_substr($row, $i - 1, 1);
                    $lc = mb_substr($row, $i + 1, 1);
                    if (ctype_alpha($fc) && ctype_alpha($lc)) {
                        self::$enclosure = '';
                    }
                }
            }
            else  self::$enclosure = '';
     
            if (self::$enclosure === false) {
                self::$errors[] = 'Unable to detect the CSV text enclosure or this file has no header';
                return false;
            } else {
                // transform header columns into array
                self::$header = explode(self::$separator, str_replace(self::$enclosure, '', trim($row, "\r\n")));
                return true;
            }
        }
    Je ne vois rien ici indiquant comment tu obtiens, pour ce fichier, les valeurs suivantes " On a les propriétés $enclosure='"' et $separator=',' "
    J'ai besoin du débug car je doute que ce soit réellement ces valeurs, justement
    Ou alors ton fichier CSV n'est pas tel que tu le crois... Comment vérifies-tu ce que ton fichier CSV contient ? Tu l'ouvres avec Excel ?

    Et l'algo me paraît tout à faire insuffisant pour détecter comment ton CSV est construit
    => Tu considères le séparateur , si ton en-tête contient au moins 1 , même s'il y a 50 ; et cela sans tenir compte du délimiteur

  11. #11
    Expert confirmé
    Avatar de laurentSc
    Homme Profil pro
    Webmaster débutant perpétuel !
    Inscrit en
    Octobre 2006
    Messages
    10 470
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Webmaster débutant perpétuel !
    Secteur : Industrie

    Informations forums :
    Inscription : Octobre 2006
    Messages : 10 470
    Points : 5 831
    Points
    5 831
    Billets dans le blog
    1
    Par défaut
    Honnêtement, j'utilise ce code codé par quelqu'un d'autre (un ancien modo d'ici).

    Pour savoir que le délimiteur est un double-quote " et le séparateur est une virgule ,, je fais un debug pas à pas avec PHPSTORM et je peux lire la valeur des variables.

    J'édite également mon fichier CSV avec cet IDE ; c'est donc ainsi que je vérifie ces valeurs.

    Comme tu n'as pas l'air d'aimer ce code, as-tu mieux à me proposer ?

Discussions similaires

  1. [C#] Changer le délimiteur d'un float
    Par trotters213 dans le forum Windows Forms
    Réponses: 2
    Dernier message: 29/03/2005, 15h59
  2. Réponses: 4
    Dernier message: 07/02/2005, 16h12
  3. Processus ininterruptible malgré IF=1
    Par imbibinebe dans le forum Assembleur
    Réponses: 3
    Dernier message: 05/12/2004, 22h08
  4. [MYSQL][deb]taille tjrs aussi grande malgre suppression
    Par floben21 dans le forum SQL Procédural
    Réponses: 3
    Dernier message: 26/08/2004, 11h39
  5. "vector" provoque "syntax error", malgré
    Par seenkay dans le forum Autres éditeurs
    Réponses: 5
    Dernier message: 24/08/2003, 03h21

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo