opencart 单入口文件简单分析

 

opencart是基于mvcl的商城系统,据说是一个外国有人单独开发。比较牛叉。但是又不大符合国人习惯,目前国内opencart社区也是不少。

简单分析了下单入口,感觉在国内商家用起来略微臃肿,需要再改进去除部分代码。

因为就在index.php上写了注释,在此贴出来,和大家一起学习。

<?php
// Version
define('VERSION', '2.1.0.1'); // 引入system初始化的配置文件,包括常量和数据库连接设置
if (is_file('config.php')) {
require_once('config.php');
} // 如果没有设置系统所需常量,需要先安装
if (!defined('DIR_APPLICATION')) {
header('Location: install/index.php');
exit;
} // Startup引入start
require_once(DIR_SYSTEM . 'startup.php');
/**
* system/startup.php做了如下事情:
*
* 1.对比php版本,magic_quotes_gpc问题,时区,判断https (ssl)问题,一般可以略过。
*
* 2.是否修正modifycation override,目前都没有修改,
直接返回system/engine下的各类。
* 注册两个自动加载自定义函数,一个就是library下的N多个类文件,
另一个是vendor下的scss(less 和scss是css的预处理器,不得不说php语言强大。。)
*
* 3.加载system/engine下的action controller event front
loader model registry类,注意传递$register的实例做为参数的类主要有
loader event customer front,目的是做一个备份,
以便处理其他controller时候处理不会丢失,
*
* 4.最后又require了helper下的几个函数
(包括生成token,处理json,utf8字符串,欧盟增值税税率扒拉扒拉,
就是那个vat文件,应该是value added tax)
*/ /**
* 接上:简单写下system/engine中各个类的作用
* 1.action.php ,在构造函数中接受路由和传递参数,
解析为controller目录下的类,excute方法new这个类并调用对应函数,
作为具体执行
*
* 2.controller 和modle为抽象类,供具体的controller和model继承使用,
注意他们都涉及魔术方法
* __get和__set,用以方便调用一个不存在的方法后调用$register里的方法
(看一下engine下的controller即可,比较简单),构造函数接受$register类。
*
* 3.loader.php 为final类,不能被继承和覆盖,
主要是做controller文件夹 model文件夹, view helper文件夹
(strup.php引入了helper啊??)
language config的一些加载,并且通过 view方法输出extract后的$data,
*
* 4.front.php 也是final类,在加载执行action之前执行
* 5.registery.php,实现简单get set,主要目的是当作参数传递给以上几个类。
* 6.event.php,还没看到具体用法,感觉像钩子,具有自定义的排序事件
*/ // Registry,全局注册类,作为参数不断的往里压入
$registry = new Registry(); // Loader,不一定需要先加载loader,需要用到$this->loader的时候会用到此类。
$loader = new Loader($registry); $registry->set('load', $loader);//此时register类里就有一个load指向了new 的loader类 // Config 这个是关键,以下会从表里把各种值都循环塞进去
$config = new Config();// 此处new的是/system/library/config.php,和register类似,但是他里面也有个load方法,load的是system/config下的文件(默认为空)
$registry->set('config', $config);// 已经有两个了!load&config // Database new 的是library下的db,(都是因为spl_autoload_register('libaray')起作用的)DB的常量是根下config的常量
$db = new DB(DB_DRIVER, DB_HOSTNAME, DB_USERNAME, DB_PASSWORD, DB_DATABASE, DB_PORT);
$registry->set('db', $db); // Store 表一 store表 判断替换 ssl和url两个字段
if (isset($_SERVER['HTTPS']) && (($_SERVER['HTTPS'] == 'on') || ($_SERVER['HTTPS'] == '1'))) {
$store_query = $db->query("SELECT * FROM " . DB_PREFIX . "store WHERE REPLACE(`ssl`, 'www.', '') = '" . $db->escape('https://' . str_replace('www.', '', $_SERVER['HTTP_HOST']) . rtrim(dirname($_SERVER['PHP_SELF']), '/.\\') . '/') . "'");
} else {// 没有https,一般走下面
$store_query = $db->query("SELECT * FROM " . DB_PREFIX . "store WHERE REPLACE(`url`, 'www.', '') = '" . $db->escape('http://' . str_replace('www.', '', $_SERVER['HTTP_HOST']) . rtrim(dirname($_SERVER['PHP_SELF']), '/.\\') . '/') . "'");
} if ($store_query->num_rows) {
$config->set('config_store_id', $store_query->row['store_id']);
} else { $config->set('config_store_id', 0);
} // Settings 表2 setting表,读取setting表的信息,
//循环所有的配置项,放在$config中,包括json格式的,
$query = $db->query("SELECT * FROM `" . DB_PREFIX . "setting` WHERE store_id = '0' OR store_id = '" . (int)$config->get('config_store_id') . "' ORDER BY store_id ASC"); foreach ($query->rows as $result) {
if (!$result['serialized']) {
// 开塞了 config
$config->set($result['key'], $result['value']);
} else {
$config->set($result['key'], json_decode($result['value'], true));
}
} if (!$store_query->num_rows) {
$config->set('config_url', HTTP_SERVER);
$config->set('config_ssl', HTTPS_SERVER);
} // Url library下的url,构造url,根据上面的config,我们常访问的?index.php?rount=a/b/c通过此类生成,传一个domain,是否ssl
$url = new Url($config->get('config_url'), $config->get('config_secure') ? $config->get('config_ssl') : $config->get('config_url')); $registry->set('url', $url); // Log 读取的setting表中的config_error_filename配置字段,默认为storage/logs/error.log
$log = new Log($config->get('config_error_filename'));
$registry->set('log', $log);
// 自定义错误函数
function error_handler($code, $message, $file, $line) {
global $log, $config; // error suppressed with @
if (error_reporting() === 0) {
return false;
} switch ($code) {
case E_NOTICE:
case E_USER_NOTICE:
$error = 'Notice';
break;
case E_WARNING:
case E_USER_WARNING:
$error = 'Warning';
break;
case E_ERROR:
case E_USER_ERROR:
$error = 'Fatal Error';
break;
default:
$error = 'Unknown';
break;
} if ($config->get('config_error_display')) {
echo '<b>' . $error . '</b>: ' . $message . ' in <b>' . $file . '</b> on line <b>' . $line . '</b>';
} if ($config->get('config_error_log')) {
$log->write('PHP ' . $error . ': ' . $message . ' in ' . $file . ' on line ' . $line);
} return true;
} // Error Handler
set_error_handler('error_handler'); // Request 请求对象,把get post server cookie 等过滤,保留clean方法对外
$request = new Request();
$registry->set('request', $request); // Response 处理header头,压缩输出,定义跳转函数,setoutput 输出数据
$response = new Response();
$response->addHeader('Content-Type: text/html; charset=utf-8');
$response->setCompression($config->get('config_compression'));
$registry->set('response', $response);

接上:貌似贴太多不能高亮~

// Cache 默认有file cache,apc cache memcache ,不过要注意,5.4以下使用apc,5.5以上使用opcache吧
// 可以参考 https://phphub.org/topics/301
$cache = new Cache('file');
$registry->set('cache', $cache); // Session 访问api接口时候,在后台系统设置-管理员-api中有体现,这个还不大明白
if (isset($request->get['token']) && isset($request->get['route']) && substr($request->get['route'], 0, 4) == 'api/') {
$db->query("DELETE FROM `" . DB_PREFIX . "api_session` WHERE TIMESTAMPADD(HOUR, 1, date_modified) < NOW()"); $query = $db->query("SELECT DISTINCT * FROM `" . DB_PREFIX . "api` `a` LEFT JOIN `" . DB_PREFIX . "api_session` `as` ON (a.api_id = as.api_id) LEFT JOIN " . DB_PREFIX . "api_ip `ai` ON (as.api_id = ai.api_id) WHERE a.status = '1' AND as.token = '" . $db->escape($request->get['token']) . "' AND ai.ip = '" . $db->escape($request->server['REMOTE_ADDR']) . "'"); if ($query->num_rows) {
// Does not seem PHP is able to handle sessions as objects properly so so wrote my own class
$session = new Session($query->row['session_id'], $query->row['session_name']);
$registry->set('session', $session); // keep the session alive
$db->query("UPDATE `" . DB_PREFIX . "api_session` SET date_modified = NOW() WHERE api_session_id = '" . $query->row['api_session_id'] . "'");
}
} else {
$session = new Session();
$registry->set('session', $session);
} // Language Detection 表四 language 多语言控制
$languages = array(); $query = $db->query("SELECT * FROM `" . DB_PREFIX . "language` WHERE status = '1'"); foreach ($query->rows as $result) {
$languages[$result['code']] = $result;
} if (isset($session->data['language']) && array_key_exists($session->data['language'], $languages)) {
$code = $session->data['language'];
} elseif (isset($request->cookie['language']) && array_key_exists($request->cookie['language'], $languages)) {
$code = $request->cookie['language'];
} else {
// detect的意思是查明,发现o(∩_∩)o
$detect = ''; // 查看请求头中有没有设置语言,
if (isset($request->server['HTTP_ACCEPT_LANGUAGE']) && $request->server['HTTP_ACCEPT_LANGUAGE']) {
$browser_languages = explode(',', $request->server['HTTP_ACCEPT_LANGUAGE']); foreach ($browser_languages as $browser_language) {
foreach ($languages as $key => $value) {
if ($value['status']) {
$locale = explode(',', $value['locale']); if (in_array($browser_language, $locale)) {
$detect = $key;
break 2;
}
}
}
}
} $code = $detect ? $detect : $config->get('config_language');
} if (!isset($session->data['language']) || $session->data['language'] != $code) {
$session->data['language'] = $code;
} if (!isset($request->cookie['language']) || $request->cookie['language'] != $code) {
setcookie('language', $code, time() + 60 * 60 * 24 * 30, '/', $request->server['HTTP_HOST']);
} $config->set('config_language_id', $languages[$code]['language_id']);
$config->set('config_language', $languages[$code]['code']); // Language
$language = new Language($languages[$code]['directory']);
$language->load($languages[$code]['directory']);
$registry->set('language', $language); // Document seo优化和获取 资源
$registry->set('document', new Document()); // Customer 获取用户信息,登录,注销等
$customer = new Customer($registry);
$registry->set('customer', $customer); // Customer Group 用户分类
if ($customer->isLogged()) {
$config->set('config_customer_group_id', $customer->getGroupId());
} elseif (isset($session->data['customer']) && isset($session->data['customer']['customer_group_id'])) {
// For API calls 批量赛
$config->set('config_customer_group_id', $session->data['customer']['customer_group_id']);
} elseif (isset($session->data['guest']) && isset($session->data['guest']['customer_group_id'])) {
$config->set('config_customer_group_id', $session->data['guest']['customer_group_id']);
} // Tracking Code
if (isset($request->get['tracking'])) {
setcookie('tracking', $request->get['tracking'], time() + 3600 * 24 * 1000, '/'); $db->query("UPDATE `" . DB_PREFIX . "marketing` SET clicks = (clicks + 1) WHERE code = '" . $db->escape($request->get['tracking']) . "'");
} // 以下new的这些都依托于 startup.php中的spl_autoload_register('library');
// Affiliate 隶属于成员,也是属于用户的信息获取,登录,注销等
$registry->set('affiliate', new Affiliate($registry)); // Currency // 货币
$registry->set('currency', new Currency($registry)); // Tax
$registry->set('tax', new Tax($registry)); // Weight
$registry->set('weight', new Weight($registry)); // Length
$registry->set('length', new Length($registry)); // Cart
$registry->set('cart', new Cart($registry)); // Encryption 加密,编码
$registry->set('encryption', new Encryption($config->get('config_encryption'))); // OpenBay Pro
$registry->set('openbay', new Openbay($registry)); // Event
$event = new Event($registry);
$registry->set('event', $event); $query = $db->query("SELECT * FROM " . DB_PREFIX . "event"); foreach ($query->rows as $result) {
$event->register($result['trigger'], $result['action']);
} // Front new 了个front,会先执行addPreAcion,(可理解为先压入的前置控制器,),然后在dispatch中具体实现,dispatch接受第二个参数为错误控制器,error/not_found.php
$controller = new Front($registry); /*new Action的作用:根据route判断是否为可用action,不可用默认为最后一个,
如果最后一个也不可用,为index 。
以下的两个addPre就是上面说的的前置控制器,可以理解为钩子吧,一个是seo,
另外的应该是(还不知道,maintenance为保养的意思,用到再说),
就是在dispatch时候,先执行前置的,在依次执行传进来的action
(通过is_callable和call_user_function )
*/ $controller->addPreAction(new Action('common/maintenance')); // SEO URL's
$controller->addPreAction(new Action('common/seo_url')); // Router 通过request对象,接受route参数,如果没有(默认首页),action为common/home就是商城首页
if (isset($request->get['route'])) {
$action = new Action($request->get['route']);
} else {
echo 'default action';
$action = new Action('common/hoe');
} /*Dispatch 派遣,调度,分发,传入一个action实例,
然后根据aciton类的构造函数得知method,然后在execute每个action
*/ $controller->dispatch($action, new Action('error/not_found')); // 在每个具体的controller最后都会执行 setoutput()方法,
//注意setoutput只接受一个参数,只不过在此是调用的load->view,通过ob得到数据 // var_dump($response->getOutput());
// 是一个string类型,最后输出 $response->output();

opencart 单入口文件简单分析的更多相关文章

  1. MVC 模型、视图、控制及其单入口文件的mvc的工作原理

         什么是mvc,mvc就是模型视图控制,模型就是model,在项目中负责数据库相关的操作,视图就是view ,负责页面的展示和数据的展示,控制就是controller ,负责中间的逻辑转换,数 ...

  2. thinkphp 的两种建构模式 第一种一个单入口里面定义两个模块,前台和后台,函数控制模块必须function.php前台加载前台模块的汉书配置文件,后台加载后台模块的汉书配置文件,公共文件共用。第二种架构模式两个单入口文件,分别生成两个应用定义define。。。函数可以定义配置文件。。。。

    thinkphp 的两种建构模式  第一种一个单入口里面定义两个模块,前台和后台,函数控制模块必须function.php前台加载前台模块的汉书配置文件,后台加载后台模块的汉书配置文件,公共文件共用. ...

  3. Android.mk文件简单分析

    Android.mk文件简单分析 一个Android.mk文件用来向编译系统描写叙述须要编译的源码.详细来说:该文件是GNUMakefile的一小部分.会被编译系统解析一次或多次. 能够在每个Andr ...

  4. webpack4 单入口文件配置 多入口文件配置 以及常用的配置

    单入口文件配置 webpack.config.js const path = require('path'); const HtmlWebpackPlugin = require('html-webp ...

  5. thinkPHP为什么设置一个单入口文件?

    TP3.2的具体解释: ThinkPHP采用单一入口模式进行项目部署和访问,无论完成什么功能,一个应用都有一个统一(但不一定是唯一)的入口. 应该说,所有应用都是从入口文件开始的,并且不同应用的入口文 ...

  6. ZooKeeper日志与快照文件简单分析

    有用过Zookeeper的都知道zoo.cfg配置文件中有dataDir配置项用于存储数据,不过可能有些人不太清楚这个目录具体存储的是那些数据,默认情况下这个目录是用于存储Log(事务日志)与Snap ...

  7. rar文件简单分析

    1.rar文件也是由许多特定的块组成 注1:CRC为CRC32的低2个字节(MARK_HEAD的CRC 为固定的0x5261,非计算出来的值) 注2: HEAD_TYPE=0x72 标记块 HEAD_ ...

  8. php对象:__autoload()函数及单入口文件,__set(), __get(), get_class_methods(),get_class_vars()

    __autoload():当类中找不到相关类的时候,会自动执行__autoload()函数,可以自动加载相关文件 __set() : 当对类的私有变量进行调用赋值时,自动调用该方法.  __get() ...

  9. thinkPHP框架单一入口文件解析

    一.index.php  (可参考ThinkPHP学习手册http://document.thinkphp.cn/manual_3_2.html#entrance_file) index.php单入口 ...

随机推荐

  1. jQuery 插件开发指南

    jQuery凭借其简洁的API,对DOM强大的操控性,易扩展性越来越受到web开发人员的喜爱,经常有人询问一些技巧,因此干脆写这么一篇文章给各位jQuery爱好者,算是抛砖引玉吧. 那么首先我们来简单 ...

  2. Sublime Text2.0.2注冊码

    // Sublime Text 3 License Keys // Sublime Text 2.x -– BEGIN LICENSE -– Andrew Weber Single User Lice ...

  3. SmartGit Mac、Liunx、Windows过期后破解方法

    根据自己的操作系统,进入相应的文件夹 ,可能还有一个版本号的文件夹,再进入 Windows: %APPDATA%\syntevo\SmartGit\ OS X: ~/Library/Preferenc ...

  4. ipc 进程间通讯的AIDL

    1.什么是aidl:aidl是 Android Interface definition language的缩写,一看就明白,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间 ...

  5. Java之线程池

    假设一个服务器完成一项任务所需时间为:T1 创建线程时间,T2 在线程中执行任务的时间,T3 销毁线程时间.当T1 + T3 远大于 T2时,采用多线程技术可以显著减少处理器单元的闲置时间,增加处理器 ...

  6. c++11——多线程

    c++11中增加了线程以及线程相关的类,很方便的支持了并发编程. 1. 线程 线程创建     使用std::thread创建线程,提供线程函数或者函数对象即可,并且可以指定线程函数的参数. #inc ...

  7. 如何用sql语句复制一张表

    如何用sql语句复制一张表 1.复制表结构及数据到新表 CREATE TABLE 新表 SELECT * FROM 旧表 这种方法会将oldtable中所有的内容都拷贝过来,当然我们可以用delete ...

  8. uva 110 Meta-Loopless Sorts 用程序写程序 有点复杂的回溯水题

    题目要求写一个直接用比较排序的pascal程序,挺有趣的一题. 我看题目数据范围就到8,本来以为贪个小便宜,用switch输出. 然后发现比较次数是阶乘级别的,8的阶乘也是挺大的,恐怕会交不上去. 于 ...

  9. 【BZOJ2982】combination Lucas定理

    [BZOJ2982]combination Description LMZ有n个不同的基友,他每天晚上要选m个进行[河蟹],而且要求每天晚上的选择都不一样.那么LMZ能够持续多少个这样的夜晚呢?当然, ...

  10. 【BZOJ2049,2631,3282,1180】LCT模板四连A

    好吧我并不想讲LCT 只是贴4个代码~ [BZOJ2049][Sdoi2008]Cave 洞穴勘测 #include <cstdio> #include <cstring> # ...