ScannedFrom Toshiba
ScannedFrom Toshiba

Reputation: 21

Import irregular XML into MySQL via PHP

i hope you can help. I have an xml file that i would like to import into a MySQL table. Unforunately it's not a standard XML file. Please see below for the format:

<?xml version="1.0" encoding="UTF-8"?>
<DDCCommonData xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" source="ECloudConnect" AuditDate="2018-11-30T13:11:32.4222315Z">
<CounterNotification>
<Date>2018-11-29T01:48:00</Date>
<MachineModel>Model 123</MachineModel>
<MachineID xsi:nil="true" />
<SerialNumber>CGD67291</SerialNumber>
<SourceDeviceID xsi:nil="true" />
<SourceModelID xsi:nil="true" />
<EncompassID xsi:nil="true" />
<ChargeCounter>
  <Counter Mode="PRINT" Color="BLACK" Type="COPY">
    <Large>0</Large>
    <Small>2043</Small>
  </Counter>
  <Counter Mode="PRINT" Color="FULL" Type="COPY">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="TWIN" Type="COPY">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="BLACK" Type="FAX">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="FULL" Type="FAX">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="TWIN" Type="FAX">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="BLACK" Type="PRINT">
    <Large>0</Large>
    <Small>2470</Small>
  </Counter>
  <Counter Mode="PRINT" Color="FULL" Type="PRINT">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="TWIN" Type="PRINT">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="BLACK" Type="LIST">
    <Large>0</Large>
    <Small>11</Small>
  </Counter>
  <Counter Mode="PRINT" Color="FULL" Type="LIST">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="PRINT" Color="TWIN" Type="LIST">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="BLACK" Type="COPY">
    <Large>0</Large>
    <Small>1270</Small>
  </Counter>
  <Counter Mode="SCAN" Color="FULL" Type="COPY">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="TWIN" Type="COPY">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="BLACK" Type="FAX">
    <Large>0</Large>
    <Small>21</Small>
  </Counter>
  <Counter Mode="SCAN" Color="FULL" Type="FAX">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="TWIN" Type="FAX">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="BLACK" Type="NET">
    <Large>1</Large>
    <Small>544</Small>
  </Counter>
  <Counter Mode="SCAN" Color="FULL" Type="NET">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
  <Counter Mode="SCAN" Color="TWIN" Type="NET">
    <Large>0</Large>
    <Small>0</Small>
  </Counter>
</ChargeCounter>
<ErrorHistory xsi:nil="true" />
<DepartmentCodes />

And i need to import the following data:

MachineModel SerialNumber

Which are the easy ones but then it gets complicated for the ChargeCounter info. I need all Large and Small data for each Counter mode:

Large Small

This is what i'm using to import the MachineModel and SerialNumber, i just need the Large and Small counters for each ChargerCounter.

<?php
$conn = mysqli_connect("localhost", "root", "test", "info");

$affectedRow = 0;

$xml = simplexml_load_file("input.xml") or die("Error: Cannot create object");

foreach ($xml->children() as $row) {
$MachineModel = $row->MachineModel ;
$SerialNumber= $row->SerialNumber;

$sql = "INSERT INTO counts(MachineModel ,SerialNumber) VALUES ('" . $MachineModel . "','" . $SerialNumber. "')";

$result = mysqli_query($conn, $sql);

if (! empty($result)) {
    $affectedRow ++;
} else {
    $error_message = mysqli_error($conn) . "\n";
}
}
?>

****UPDATE****

I need the output to fit this table:

CREATE TABLE info (
id int(9) NOT NULL auto_increment,

modelname varchar(255) NULL,
serialnumber varchar(255) NULL,

print_black_copy_sm varchar(255) NULL,
print_black_copy_lg varchar(255) NULL,
print_full_copy_sm varchar(255) NULL,
print_full_copy_lg varchar(255) NULL,
print_twin_copy_sm varchar(255) NULL,
print_twin_copy_lg varchar(255) NULL,

print_black_fax_sm varchar(255) NULL,
print_black_fax_lg varchar(255) NULL,
print_full_fax_sm varchar(255) NULL,
print_full_fax_lg varchar(255) NULL,
print_twin_fax_sm varchar(255) NULL,
print_twin_fax_lg varchar(255) NULL,

print_black_print_sm varchar(255) NULL,
print_black_print_lg varchar(255) NULL,
print_full_print_sm varchar(255) NULL,
print_full_print_lg varchar(255) NULL,
print_twin_print_sm varchar(255) NULL,
print_twin_print_lg varchar(255) NULL,

print_black_list_sm varchar(255) NULL,
print_black_list_lg varchar(255) NULL,
print_full_list_sm varchar(255) NULL,
print_full_list_lg varchar(255) NULL,
print_twin_list_sm varchar(255) NULL,
print_twin_list_lg varchar(255) NULL,

scan_black_copy_sm varchar(255) NULL,
scan_black_copy_lg varchar(255) NULL,
scan_full_copy_sm varchar(255) NULL,
scan_full_copy_lg varchar(255) NULL,
scan_twin_copy_sm varchar(255) NULL,
scan_twin_copy_lg varchar(255) NULL,

scan_black_fax_sm varchar(255) NULL,
scan_black_fax_lg varchar(255) NULL,
scan_full_fax_sm varchar(255) NULL,
scan_full_fax_lg varchar(255) NULL,
scan_twin_fax_sm varchar(255) NULL,
scan_twin_fax_lg varchar(255) NULL,

scan_black_net_sm varchar(255) NULL,
scan_black_net_lg varchar(255) NULL,
scan_full_net_sm varchar(255) NULL,
scan_full_net_lg varchar(255) NULL,
scan_twin_net_sm varchar(255) NULL,
scan_twin_net_lg varchar(255) NULL,

PRIMARY KEY (id)
) ENGINE=InnoDB;

Upvotes: 0

Views: 63

Answers (2)

Nigel Ren
Nigel Ren

Reputation: 57121

Your table structure is very badly designed, storing repeated data like this is not recommended, instead I would suggest something like...

CREATE TABLE counts (
    id int(9) NOT NULL auto_increment,
    modelname varchar(255) NULL,
    serialnumber varchar(255) NULL,
    counter_color varchar(255) NULL,
    counter_type varchar(255) NULL,
    counter_small varchar(255) NULL,
    counter_large varchar(255) NULL
    PRIMARY KEY (id)
) ENGINE=InnoDB;

You can then read each counter element and using the Color and Type attributes, store it directly into the table rather than having 1 huge table...

$xml = simplexml_load_file("NewFile.xml") or die("Error: Cannot create object");

$sql = "INSERT INTO counts(MachineModel ,SerialNumber, counter_color, counter_type
            counter_small, counter_large) 
    VALUES (?,?,?,?,?,?)";
$stmt = mysqli_prepare($conn, $sql) or die(mysqli_error($conn));

foreach ($xml->CounterNotification as $row) {
    $MachineModel = $row->MachineModel ;
    $SerialNumber= $row->SerialNumber;

    echo $MachineModel."/".$SerialNumber.PHP_EOL;
    foreach ( $row->ChargeCounter->Counter as $counter )    {
        mysqli_stmt_bind_param($stmt, "ssssss", $MachineModel, $SerialNumber,
            $counter['Color'], $counter['Type'], $counter->Small, $counter->Large);
        $result = mysqli_execute($stmt);
        if (! empty($result)) {
            $affectedRow ++;
        } else {
            $error_message = mysqli_error($conn) . "\n";
        }
    }
}

Upvotes: 0

jbrahy
jbrahy

Reputation: 4415

Here's a hack. It's just basically completing the XML file.

<?php
$conn = mysqli_connect("localhost", "root", "test", "info");

$affectedRow = 0;

$file_contents = sprintf("%s</CounterNotification> </DDCCommonData>",file_get_contents("input.xml"));

$xml = simplexml_load_string($file_contents) or die("Error: Cannot create object");

foreach ($xml->children() as $row) {
    $MachineModel = $row->MachineModel ;
    $SerialNumber= $row->SerialNumber;

    $sql = "INSERT INTO counts(MachineModel ,SerialNumber) VALUES ('" . $MachineModel . "','" . $SerialNumber. "')";

    $result = mysqli_query($conn, $sql);

    if (! empty($result)) {
        $affectedRow ++;
    } else {
        $error_message = mysqli_error($conn) . "\n";
    }
}

Upvotes: 1

Related Questions