-
Notifications
You must be signed in to change notification settings - Fork 114
Module.user
Taiwen Jiang edited this page Aug 15, 2013
·
10 revisions
# Pi Engine schema
# http://pialog.org
# Author: Taiwen Jiang <[email protected]>
# --------------------------------------------------------
# ------------------------------------------------------
# User
# >>>>
# user ID: unique in the system, referenced as `uid`
# user identity: unique identity, generated by system or set by third-pary
# all local data of a user should be indexed by user ID
# User account and authentication data
CREATE TABLE `{account}` (
`id` int(10) unsigned NOT NULL auto_increment,
`identity` varchar(32) NOT NULL,
`credential` varchar(255) NOT NULL default '', # Credential hash
`salt` varchar(255) NOT NULL default '', # Hash salt
`email` varchar(64) NOT NULL,
`name` varchar(255) NOT NULL,
`active` tinyint(1) unsigned NOT NULL default '0',
`time_created` int(10) unsigned NOT NULL default '0',
`time_activated` int(10) unsigned NOT NULL default '0',
`time_disabled` int(10) unsigned NOT NULL default '0',
`time_deleted` int(10) unsigned NOT NULL default '0',
PRIMARY KEY (`id`),
UNIQUE KEY `identity` (`identity`),
UNIQUE KEY `email` (`email`),
UNIQUE KEY `name` (`name`),
KEY `status` (`active`)
);
# Profile schema for basic fields
CREATE TABLE `{profile}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`gender` enum('male', 'female', 'unknown'),
`fullname` varchar(255) NOT NULL default '',
`birthdate` varchar(10) NOT NULL default '', # YYYY-mm-dd
`location` varchar(255) NOT NULL default '',
`signature` varchar(255) NOT NULL default '',
`avatar` varchar(255) NOT NULL default '',
`bio` text,
PRIMARY KEY (`id`),
UNIQUE KEY `user` (`uid`)
);
# Entity meta for custom user profile fields
CREATE TABLE `{custom}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`field` varchar(64) NOT NULL, # Custom field name
`value` text,
PRIMARY KEY (`id`),
UNIQUE KEY `field` (`uid`, `field`)
);
# Entity for user profile compound fields
CREATE TABLE `{compound}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`compound` varchar(64) NOT NULL, # Compound name
`set` varchar(64) default NULL, # Field set key
`field` varchar(64) NOT NULL, # Compound field name
`value` text,
PRIMARY KEY (`id`),
UNIQUE KEY `field` (`uid`, `compound`, `set`, `field`)
);
# Entity meta for all profile fields: account, basic profile and custom fields
CREATE TABLE `{field}` (
`id` smallint(5) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL,
`module` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`edit` text, # callback options for edit
`filter` text, # callback options for output filtering
`type` enum('custom', 'account', 'profile', 'compound'), # Field type, default as custom
`is_edit` tinyint(1) NOT NULL default '0', # Is editable by user
`is_search` tinyint(1) NOT NULL default '0', # Is searchable
`is_display` tinyint(1) NOT NULL default '0', # Display on profile page
`active` tinyint(1) NOT NULL default '0', # Is active
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`name`)
);
# Entity meta for compound fields
CREATE TABLE `{compound_field}` (
`id` smallint(5) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL,
`compound` varchar(64) NOT NULL default '',
`module` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`edit` text, # callback options for edit
`filter` text, # callback options for output filtering
`active` tinyint(1) NOT NULL default '0', # Is active
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`compound`, `name`)
);
# Display group for profile fields
CREATE TABLE `{display_group}` (
`id` int(10) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`order` smallint(5) unsigned NOT NULL default '0',
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`name`)
);
# Display grouping and order of field
CREATE TABLE `{field_display}` (
`id` int(10) unsigned NOT NULL auto_increment,
`field` varchar(64) NOT NULL default '',
`group` varchar(64) NOT NULL default '',
`order` smallint(5) unsigned NOT NULL default '0',
PRIMARY KEY (`id`)
);
# Timeline meta
CREATE TABLE `{timeline}` (
`id` int(10) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`module` varchar(64) NOT NULL default '',
`icon` varchar(255) NOT NULL default '',
`active` tinyint(1) NOT NULL default '0', # Is active
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`module`, `name`)
);
# Activity meta
CREATE TABLE `{activity}` (
`id` int(10) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`callback` varchar(64) NOT NULL,
`module` varchar(64) NOT NULL default '',
`link` varchar(255) NOT NULL default '',
`icon` varchar(255) NOT NULL default '',
`active` tinyint(1) NOT NULL default '0', # Is active
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`module`, `name`)
);
# Quicklinks
CREATE TABLE `{quicklink}` (
`id` int(10) unsigned NOT NULL auto_increment,
`name` varchar(64) NOT NULL default '',
`title` varchar(255) NOT NULL default '',
`module` varchar(64) NOT NULL default '',
`link` varchar(255) NOT NULL default '',
`icon` varchar(255) NOT NULL default '',
`active` tinyint(1) NOT NULL default '0', # Is active
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`module`, `name`)
);
# Timeline for user activities
CREATE TABLE `{timeline_log}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`timeline` varchar(64) NOT NULL default '',
`module` varchar(64) NOT NULL default '',
`message` text,
`link` varchar(255) NOT NULL default '',
`time` int(11) unsigned NOT NULL,
PRIMARY KEY (`id`),
KEY (`uid`)
);
# User action log
CREATE TABLE `{log}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`time` int(10) unsigned NOT NULL default '0',
`data` varchar(255) NOT NULL default '',
`action` varchar(64) NOT NULL default '',
`module` varchar(64) NOT NULL default '',
PRIMARY KEY (`id`),
KEY (`uid`)
);
# ------------------------------------------------------
# user custom contents
CREATE TABLE `{repo}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL default '0',
`module` varchar(64) NOT NULL default '',
`type` varchar(64) NOT NULL default '',
`content` text,
PRIMARY KEY (`id`),
UNIQUE KEY `user_content` (`uid`, `module`, `type`)
);
# user-role links for regular
CREATE TABLE `{role}` (
`id` int(10) unsigned NOT NULL auto_increment,
`uid` int(10) unsigned NOT NULL,
`role` varchar(64) NOT NULL default '',
`section` varchar(64) NOT NULL default 'front',
PRIMARY KEY (`id`),
UNIQUE KEY `user` (`section`, `uid`)
);
- Field name will be defined as _ if not specified
- The entities will be inserted into
field
table withtype
as account or profile - User account data and profile data for the fields will be inserted into account and profile respectively
array(
'account' => array(
<field-key> => array(
// Field name: if the 'name' is not specified, field name will be defined as '<module-name>_<field-key>'
'name' => <field-name>
// Title
'title' => <Field Title>
// Edit element specs
'edit' => <form-element-filter-validator>,
// Filter for value processing for output
'filter' => <output-filter>
// Is editable by admin, default as true
'is_edit' => <bool>,
// Display on user profile page, default as true
'is_display' => <bool>,
// Search user by this field, default as true
'is_search' => <bool>,
),
<...>,
),
'profile' => array(
<field-key> => array(
// Field name: if the 'name' is not specified, field name will be defined as '<module-name>_<field-key>'
'name' => <field-name>
// Title
'title' => <Field Title>
// Edit element specs
'edit' => <form-element-filter-validator>,
// Filter for value processing for output
'filter' => <output-filter>
// Is editable by admin, default as true
'is_edit' => <bool>,
// Display on user profile page, default as true
'is_display' => <bool>,
// Search user by this field, default as true
'is_search' => <bool>,
),
<...>,
),
),
- Compound name will be defined as _ if not specified
- Compound entities will be inserted into
field
table withtype
as compound - Compound field name will be defined as -
- Fields of compounds will be inserted into
compound_field
table - User data for the compound entities will be inserted into
compound
table
// Social networking tools
'tool' => array(
'name' => 'tool',
'title' => __('Social tools'),
'table' => 'tool',
'field' => array(
'title' => array(
'title' => __('Tool name'),
),
'identifier' => array(
'title' => __('ID or URL'),
),
),
),
// Communication address
'address' => array(
'name' => 'address',
'title' => __('Address'),
'table' => 'address',
'field' => array(
'postcode' => array(
'title' => __('Post code'),
),
'country' => array(
'title' => __('Country'),
),
'province' => array(
'title' => __('Province'),
),
'city' => array(
'title' => __('City'),
),
'street' => array(
'title' => __('Street'),
),
'office' => array(
'title' => __('Office'),
),
),
),
// Education experiences
'education' => array(
'name' => 'education',
'title' => __('Education'),
'table' => 'education',
'field' => array(
'school' => array(
'title' => __('School name'),
),
'major' => array(
'title' => __('Major'),
),
'degree' => array(
'title' => __('Degree'),
),
'class' => array(
'title' => __('Class'),
),
'start' => array(
'title' => __('Start time'),
),
'end' => array(
'title' => __('End time'),
),
),
),
// Profession experiences
'work' => array(
'name' => 'work',
'title' => __('Work'),
'table' => 'work',
'field' => array(
'company' => array(
'title' => __('Company name'),
),
'department' => array(
'title' => __('Department'),
),
'title' => array(
'title' => __('Job title'),
),
'description' => array(
'title' => __('Description'),
'edit' => 'textarea',
),
'start' => array(
'title' => __('Start time'),
),
'end' => array(
'title' => __('End time'),
),
),
),
- Load
field
fromextra/user/config/compound.php
if file available, otherwise - Load
field
frommodule/user/config/compound.php
- The entities will be inserted into
field
table with:- Field name will be defined as _ if not specified
- Field type is set:
type
as custom
- User profile data for active fields will be inserted into
custom
table
array(
'field' => array(
<field-key> => array(
// Field name: if the 'name' is not specified, field name will be defined as '<module-name>_<field-key>'
'name' => <field-name>
// Title
'title' => <Field Title>
// Edit element specs
'edit' => <form-element-filter-validator>,
// Filter for value processing for output
'filter' => <output-filter>
// Is editable, default as true
'is_edit' => <bool>,
// Display on user profile page, default as true
'is_display' => <bool>,
// Search user by this field, default as true
'is_search' => <bool>,
),
<...>,
),
),
- Load
field
fromextra/user/config/user.php
if file available, otherwise - Load
field
frommodule/user/config/user.php
use Module\User\Api\User as UserApi;
- Call by
UserApi::canonizeMeta($rawData, $action = null)
- Return array for account, profile, custom
$result = array(
'account' => array(),
'profile' => array(),
'custom' => array(),
);
foreach ($result as $type => &$data) {
$fields = Pi::service('user')->getMeta($type, $action);
foreach ($fields as $field) {
if (in_array($field, $rawData)) {
$data[] = $field;
}
}
}
return $result;
- Call by
UserApi::canonizeUser($rawData)
- Return associative array of account, profile, custom
$result = array(
'account' => array(),
'profile' => array(),
'custom' => array(),
);
foreach ($result as $type => &$data) {
$fields = Pi::service('user')->getMeta($type);
foreach ($fields as $field) {
if (array_key_exists($field, $rawData)) {
$data[$key][$field] = $rawData[$field];
}
}
}
return $result;
- Call by
UserApi::add(array $data)
return<uid>
$data = UserApi::canonizeUser($data);
$uid = UserApi::addAccount($data['account']);
$id = UserApi::addProfile($data['profile'], $uid);
$results = UserApi::addCustom($data['custom'], $uid);
return $uid;
- Add user account
UserApi::addAccount(array $account)
return<uid>
$row = Pi::model('account', 'user')->createRow($account);
$uid = $row->save();
return $uid;
- Add user profile
UserApi::addProfile(array $profile, $uid)
return<profile-id>
$profile['uid'] = $uid;
$row = Pi::model('profile', 'user')->createRow($profile);
$id = $row->save();
return $id;
- Add user custom entities
UserApi::addCustom($custom, $uid)
return<bool[]>
$result = array();
foreach (custom as $key => $value) {
$id = UserApi::addEntity($key, $value, $uid);
$result[$key] = $id;
}
return $result;
- Add user custom entity
UserApi::addEntity($name, $value, $uid)
return<entity-id>
$entity = array(
'uid' => $uid,
'field' => $name,
'value' => $value,
);
$row = Pi::model('custom', 'user')->createRow($entity);
$id = $row->save();
return $id;
- Call by
UserApi::update(array $data, $uid)
return<bool>
$data = UserApi::canonizeUser($data);
$status = UserApi::updateAccount($data['account'], $uid);
$status = UserApi::updateProfile($data['profile'], $uid);
$status = UserApi::updateCustom($data['custom'], $uid);
return $status;
- Update user account
UserApi::updateAccount(array $account, $uid)
return<bool>
$status = Pi::model('account', 'user')->update($account, array('id' => $uid));
- Update user profile
UserApi::updateProfile(array $profile, $uid)
return<bool>
$status = Pi::model('profile', 'user')->update($profile, array('uid' => $uid));
- Update user custom profile
UserApi::updateCustom(array $custom, $uid)
return<bool>
foreach (custom as $key => $value) {
$status = UserApi::updateEntity($key, $value, $uid);
}
- Update user custom entity
UserApi::updateEntity($name, $value, $uid)
return<bool>
$status = Pi::model('custom', 'user')->update(
array('value' => $value),
array(
'field' => $name,
'uid' => $uid,
)
);
- Call by
UserApi::delete($uid)
return<bool>
$status = UserApi::deleteAccount($uid);
$status = UserApi::deleteProfile($uid);
$status = UserApi::deleteCustom($uid);
return $status;
- Delete user account
UserApi::deleteAccount($uid)
return<bool>
$status = Pi::model('account', 'user')->delete(array('id' => $uid));
- Delete user profile
UserApi::deleteProfile($uid)
return<bool>
$status = Pi::model('profile', 'user')->delete(array('uid' => $uid));
- Delete user custom profile
UserApi::deleteCustom($uid)
return<bool>
$status = Pi::model('custom', 'user')->delete(array('uid' => $uid));
- Delete user custom entity
UserApi::deleteEntity($name, $uid)
return<bool>
$status = Pi::model('custom', 'user')->delete(
array(
'field' => $name,
'uid' => $uid,
)
);
- Call by
UserApi::get($uid, $action = null, array $fields = null)
return associative array
$account = array();
$profile = array();
$custom = array();
$account = UserApi::getAccount($uid, $action, $fields);
$profile = UserApi::getProfile($uid, $action, $fields);
$Custom = UserApi::getCustom($uid, $action, $fields);
$result = $account + $profile + $custom;
- Get user account
UserApi::getAccount($uid, $action = null, array $fields = null)
$type = 'account';
$model = Pi::model('account', 'user');
$where = array('id' = $uid);
$result = array();
$columns = Pi::registry('profile', 'user')->read($type, $action);
if ($fields) {
foreach (array_keys($columns) as $field) {
if (!in_array($field, $fields)) {
unset(columns[$field]);
}
}
}
if (!$columns) {
return $result;
}
$select = $model->select()->where($where)->columns($columns);
$rowset = $model->selectWith($select);
$result = $rowset->current()->toArray();
if ('display' == $action) {
foreach ($result as $key => &$value) {
if ($columns[$key]['filter']) {
$value = call_user_func($columns[$key]['filter'], $value);
}
}
}
- Get user profile
UserApi::getProfile($uid, $action = null, array $fields = null)
$type = 'account';
$model = Pi::model('account', 'user');
$where = array('uid' = $uid);
// Same as account
<...>
- Get user custom profile
UserApi::getCustom($uid, $action = null, array $fields = null)
$type = 'custom';
$model = Pi::model('custom', 'user');
$where = array('uid' = $uid);
$result = array();
$columns = Pi::registry('profile', 'user')->read($type, $action);
if ($fields) {
foreach (array_keys($columns) as $field) {
if (!in_array($field, $fields)) {
unset(columns[$field]);
}
}
}
if (!$columns) {
return $result;
}
$where['field'] = array_keys($columns);
$select = $model->select()->where($where)->columns($columns);
$rowset = $model->selectWith($select);
foreach ($rowset as $row) {
$result[$row->field] = $row->value;
}
if ('display' == $action) {
foreach ($result as $key => &$value) {
if ($columns[$key]['filter']) {
$value = call_user_func($columns[$key]['filter'], $value);
}
}
}
- Call by
UserApi::getField($uid, $field, $action = '', $type = '')
$columns = Pi::registry('profile', 'user')->read($type, $action);
if (!isset($columns[$field])) {
return null;
}
$type = $type ?: $columns[$field]['type];
$method = 'get' . ucfirst($type);
$data = UserApi::{$method}($uid, (array) $field, $action);
if (isset($data[$field])) {
$result = $data[$field];
} else {
$result = null;
}
- Call by
Pi::registry('profile', 'user')->read(<type>, <action>)
return array of fields asstring[]
- Type: account, profile, custom
- action: display, edit, search
$columns = array();
$where = array('active' => 1);
if ($type) {
$where['type'] = $type;
}
switch ($action) {
case 'display':
$columns = array('name', 'title', 'filter');
$where['is_display'] = 1;
break;
case 'edit':
$columns = array('name', 'title', 'edit');
$where['is_edit'] = 1;
break;
case 'search':
$columns = array('name', 'title');
$where['is_search'] = 1;
break;
default:
break;
}
$select = Pi::model('field', 'user')->select()->where($where);
if ($columns) {
$select->columns($columns);
}
$rowset = Pi::model('field', 'user')->selectWith($select);
foreach ($rowset as $row) {
$result[$row->name] = $row->toArray();
}
- Identified by
is_edit
attribute
$fields = Pi::registry('profile', 'user')->read('', 'edit');
- Identified by non-empty
edit
attribute
$fields = Pi::registry('profile', 'user')->read();
foreach ($fields as $key => $data) {
if (!$data['edit']) {
unset($fields[$key]);
}
}
$user = array(
'uid' => $id,
);
$registryService = Pi::registry('profile', 'user');
$fields = $registryService->read('account');
$row = Pi::model('account', 'user')->find($id);
$account = array(
'id' => $row->id,
);
foreach (array_keys($fields) as $key) {
$account[$key] = $row->{$key};
}
$user += $account;
$fields = $registryService->read('profile');
$row = Pi::model('profile', 'user')->find($id);
$profile = array(
'id' => $row->id,
'uid' => $row->uid,
);
foreach (array_keys($fields) as $key) {
$profile[$key] = $row->{$key};
}
$user += $profile;
$fields = $registryService->read('custom');
$rowset = Pi::model('custom', 'user')->select(array(
'uid' => $uid,
'field' => array_keys($fields),
));
$custom = array(
'uid' => $row->uid,
);
foreach ($rowset as $row) {
$custom[$row->field] = $row->value;
}
$user += $custom;