csv names * @var array */ protected $fieldMap = []; /** * @var callable */ protected $logCallback; protected $tempPassword; /** * @var callable */ protected $progressCallback; /** * @var null */ protected $usernameFormat; /** * @var callable */ protected $errorCallback; /** * ObjectImporter constructor. * @param string $filename */ public function __construct($filename) { $this->filename = $filename; $this->csv = Reader::createFromPath($filename); $this->csv->setNewLine('\r\n'); if (! ini_get("auto_detect_line_endings")) { ini_set("auto_detect_line_endings", '1'); } $this->tempPassword = substr(str_shuffle("0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"), 0, 20); } // Cached Values for import lookups protected $customFields; public function import() { $results = $this->csv->fetchAssoc(); $this->customFields = CustomField::All(['name']); DB::transaction(function () use (&$results) { Model::unguard(); $resultsCount = sizeof($results); foreach ($results as $row) { $this->handle($row); call_user_func($this->progressCallback, $resultsCount); $this->log('------------- Action Summary ----------------'); } }); } abstract protected function handle($row); /** * Check to see if the given key exists in the array, and trim excess white space before returning it * * @author Daniel Melzter * @since 3.0 * @param $array array * @param $key string * @param $default string * @return string */ public function findCsvMatch(array $array, $key, $default = '') { $val = $default; // dd($array); if($customKey = $this->lookupCustomKey($key)) { $key = $customKey; } $this->log("Custom Key: ${key}"); if (array_key_exists($key, $array)) { $val = e(Encoding::toUTF8(trim($array[ $key ]))); } // $this->log("${key}: ${val}"); return $val; } /** * Looks up A custom key in the custom field map * * @author Daniel Melzter * @since 4.0 * @param $key string * @return string|null */ public function lookupCustomKey($key) { // dd($this->fieldMap); if (array_key_exists($key, $this->fieldMap)) { // $this->log("Found a match in our custom map: {$key} is " . $this->fieldMap[$key]); return $key = $this->fieldMap[$key]; } return null; } /** * Figure out the fieldname of the custom field * * @author A. Gianotto * @since 3.0 * @param $array array * @return string */ public function array_smart_custom_field_fetch(array $array, $key) { $index_name = strtolower($key->name); return array_key_exists($index_name, $array) ? e(trim($array[$index_name])) : false; } protected function log($string) { call_user_func($this->logCallback, $string); } protected function logError($item, $field) { call_user_func($this->errorCallback, $item, $field, $item->getErrors()); } /** * Finds the user matching given data, or creates a new one if there is no match * * @author Daniel Melzter * @since 3.0 * @param $row array * @return User Model w/ matching name * @internal param string $user_username Username extracted from CSV * @internal param string $user_email Email extracted from CSV * @internal param string $first_name * @internal param string $last_name */ protected function createOrFetchUser($row) { $user_name = $this->findCsvMatch($row, "name"); $user_email = $this->findCsvMatch($row, "email"); $user_username = $this->findCsvMatch($row, "username"); $first_name = ''; $last_name = ''; // A number was given instead of a name if (is_numeric($user_name)) { $this->log('User '.$user_name.' is not a name - assume this user already exists'); $user_username = ''; // No name was given } elseif (empty($user_name)) { $this->log('No user data provided - skipping user creation, just adding asset'); //$user_username = ''; } else { $user_email_array = User::generateFormattedNameFromFullName(Setting::getSettings()->email_format, $user_name); $first_name = $user_email_array['first_name']; $last_name = $user_email_array['last_name']; if ($user_email=='') { if (Setting::getSettings()->email_domain) { $user_email = str_slug($user_email_array['username']).'@'.Setting::getSettings()->email_domain; } } if ($user_username=='') { if ($this->usernameFormat =='email') { $user_username = $user_email; } else { $user_name_array = User::generateFormattedNameFromFullName(Setting::getSettings()->username_format, $user_name); $user_username = $user_name_array['username']; } } } $user = new User; if ($this->testRun) { return $user; } if (!empty($user_username)) { if ($user = User::MatchEmailOrUsername($user_username, $user_email) ->whereNotNull('username')->first()) { $this->log('User '.$user_username.' already exists'); } elseif (( $first_name != '') && ($last_name != '') && ($user_username != '')) { $user = new User; $user->first_name = $first_name; $user->last_name = $last_name; $user->username = $user_username; $user->email = $user_email; $user->activated = 1; $user->password = $this->tempPassword; if ($user->save()) { $this->log('User '.$first_name.' created'); } else { $this->logError($user, 'User "' . $first_name . '"'); } } } return $user; } /** * Sets the value of filename. * * @param string $filename the filename * * @return self */ public function setFilename($filename) { $this->filename = $filename; return $this; } /** * Sets the Should we persist to database?. * * @param bool $testRun the test run * * @return self */ public function setTestRun($testRun) { $this->testRun = $testRun; return $this; } /** * Sets the Id of User performing import. * * @param mixed $user_id the user id * * @return self */ public function setUserId($user_id) { $this->user_id = $user_id; return $this; } /** * Sets the Are we updating items in the import. * * @param bool $updating the updating * * @return self */ public function setUpdating($updating) { $this->updating = $updating; return $this; } /** * Defines mappings of csv fields * * @param bool $updating the updating * * @return self */ public function setFieldMappings($fields) { // Some initial sanitization. $this->fieldMap = $fields; $this->log($this->fieldMap); return $this; } /** * Sets the callbacks for the import * * @param callable $logCallback Function to call when we have data to log * @param callable $progressCallback Function to call to display progress * @param callable $errorCallback Function to call when we have errors * * @return self */ public function setCallbacks(callable $logCallback, callable $progressCallback, callable $errorCallback) { $this->logCallback = $logCallback; $this->progressCallback = $progressCallback; $this->errorCallback = $errorCallback; return $this; } /** * Sets the value of usernameFormat. * * @param string $usernameFormat the username format * * @return self */ public function setUsernameFormat($usernameFormat) { $this->usernameFormat = $usernameFormat; return $this; } }