onlyphp / codeigniter3-csvimporter
A robust CSV importer library for CodeIgniter 3 with background processing support, progress tracking, and detailed statistics. Perfect for handling large CSV files without timeouts or memory issues.
Requires
- php: >=8.0
- opis/closure: *
README
A robust CSV importer library for CodeIgniter 3 with background processing support, progress tracking, and detailed statistics. Perfect for handling large CSV files without timeouts or memory issues.
⚠️ Warning
DO NOT USE THIS PACKAGE IN PRODUCTION
This package is under active development and may contain critical bugs. It is primarily intended for personal use and testing. The current version has not undergone rigorous testing and may be unstable.
✨ Features
- 🚀 Background processing support with OS-specific optimizations
- 📊 Real-time progress tracking
- 📈 Detailed statistics (inserts/updates/errors)
- ⚠️ Comprehensive error handling
- 💻 Shared hosting compatible & Cross-platform compatible (Windows & Linux)
- ⏰ No cron job required
- 🛠️ Customizable processing logic
- 🔄 Memory-efficient chunk processing
- ⚡ Configurable processing parameters
- 📝 Skip empty rows automatically
- 🔍 Detailed error tracking
- 🖥️ Smart CPU load management
- 🔄 Automatic process recovery
- 🛡️ Process locking mechanism
🔧 System Requirements
- PHP 8.0 or higher
- CodeIgniter 3.x
proc_open
andproc_close
PHP functions enabledMySQL
database- Write permissions for temporary directory
- For Linux:
mpstat
command available for CPU monitoring - For Windows:
wmic
command available for CPU monitoring
📦 Installation
Install via Composer:
composer require onlyphp/codeigniter3-csvimporter
📝 Usage
Basic Usage
// Initialize the processor $processor = new \OnlyPHP\CSVSimpleImporter\CSVImportProcessor(); // Set callback function for processing each row $processor->setCallback(function($row, $rowIndex, $models) { try { // Process your row data here return [ 'code' => 200, 'action' => 'create', 'message' => 'Success' ]; } catch (\Exception $e) { return [ 'code' => 500, 'error' => $e->getMessage() ]; } }); // Start processing $jobId = $processor->process('/path/to/your/file.csv');
Advanced Configuration
$processor = new \OnlyPHP\CSVSimpleImporter\CSVImportProcessor(); // Set user ID for file ownership $processor->setFileBelongsTo(1); // Set HTML element ID for frontend progress tracking $processor->setDisplayHTMLId('progress-bar-1'); // Configure CSV processing parameters $processor->setMemoryLimit('1G') ->setDelimiter(',') ->setEnclosure('"') ->setEscape('\\') ->setChunkSize(1000) ->setRecordUpdateInterval(250) ->setSkipHeader(true); // Load specific models for processing $processor->setCallbackModel(['User_model', 'Product_model']); // Set callback with loaded models $processor->setCallback(function($row, $rowIndex, $models) { $userModel = $models['User_model']; $productModel = $models['Product_model']; try { // Your processing logic here $result = $userModel->createFromCSV($row); return [ 'code' => 200, 'action' => 'create', 'message' => 'User created successfully' ]; } catch (\Exception $e) { return [ 'code' => 500, 'error' => 'Row ' . $rowIndex . ': ' . $e->getMessage() ]; } }); // Start processing $jobId = $processor->process('/path/to/your/encryptFileName.csv', 'originalFileName.csv');
Configuration Options
Process Control
// Kill a running process $processor->killProcess($jobId); // Check process status $status = $processor->getStatus($jobId); // Check status for all processes owned by a user $status = $processor->getStatusByOwner($userId);
Status Response Format
[ 'job_id' => 'csv_123456789', 'total_process' => 100, 'total_success' => 95, 'total_failed' => 5, 'total_inserted' => 80, 'total_updated' => 15, 'total_skip_empty_row' => 3, 'display_id' => 'progress-bar-1', 'estimate_time' => [ 'hours' => 0, 'minutes' => 5, 'seconds' => 30 ], 'file_name' => 'users.csv', 'status' => 2, // 1=Pending, 2=Processing, 3=Completed, 4=Failed 'error_message' => '[]', 'percentage_completion' => '10', 'last_check' => '2024-01-05 12:34:56' ]
Frontend Integration Example
function checkProgress(jobId) { $.ajax({ url: "/your-controller/check-progress", data: { job_id: jobId }, success: function (response) { if (response.status == 3) { // Process completed $("#progress-bar-1").html("Import completed!"); } else if (response.status == 4) { // Process failed $("#progress-bar-1").html("Import failed: " + response.error_message); } else { // Update progress let progress = response.percentage_completion; $("#progress-bar-1").css("width", progress + "%"); $("#progress-bar-1").html(`${progress}% (${response.total_success} succeeded, ${response.total_failed} failed)`); // Check again in 2 seconds setTimeout(() => checkProgress(jobId), 2000); } }, }); }
📊 Status Codes Reference
⚙️ Callback Response Format
The callback function should return an array with the following structure:
// Success response return [ 'code' => 200, 'action' => 'create', // or 'update' 'message' => 'Success message' ]; // Error response return [ 'code' => 500, 'error' => 'Error message' ];
🔒 Process Management Features
CPU Load Management
The system automatically monitors server CPU load and manages processes accordingly:
- Delays process start if CPU load is above 90%
- Continuously monitors load during processing
- Platform-specific CPU monitoring (Linux uses
mpstat
, Windows useswmic
)
Process Recovery
- Automatic cleanup of orphaned processes
- Lock file management to prevent duplicate processing
- Graceful handling of interrupted processes
Memory Management
- Chunk-based processing to control memory usage
- Configurable memory limits
- Automatic garbage collection
- Database connection management to prevent leaks
🤝 Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
- Fork the Project
- Create your Feature Branch (
git checkout -b feature/AmazingFeature
) - Commit your Changes (
git commit -m 'Add some AmazingFeature'
) - Push to the Branch (
git push origin feature/AmazingFeature
) - Open a Pull Request
📄 License
The MIT License (MIT). Please see License File for more information.
💖 Support
If you find this library helpful, please consider giving it a star on GitHub!