Using Zend_Auth to secure all controllers

后端 未结 4 381
醉梦人生
醉梦人生 2021-01-02 23:15

How would i globally secure all my controllers (except for my Login controller) to ensure my application is secure at all points (no hidden backdoor to ajax calls, etc). I t

4条回答
  •  别那么骄傲
    2021-01-02 23:46

    edit: this is a complement of @singles response.

    You must understand there are 2 different things. Auth and Acl. Auth tells you who is the user, and you can for example redirect user having no Auth to you login controller, and set an auth identity after login. then the Acl system take yes/no decisions based on the Auth data (could be the user id or is role, stored in the Auth storage.

    On nice solution is to have 2 controllers plugins (registered in the good order on the bootstrap, Auth then Acl). If you do not use Controller plugins you'll have to call the Acl check in each controller, when needed. If you always need it, then use plugins.

    Implement the preDispatch() in you Auth plugin to set for example an anonymous identity if you have no identity return from Zend_Auth. This is a code snippet of a real one:

    public function preDispatch(Zend_Controller_Request_Abstract $request)
    {
        $module = $request->getModuleName();
        $controller = $request->getControllerName();
        $action = $request->getActionName();
        $auth = Zend_Auth::getInstance();
        if (!$auth->hasIdentity()) {
            // set a default anonymous identity
            $auth->getStorage()->write(array('name' => 'anonymous','role' => 1,));
        }
    (...)
    

    And for the Acl controller plugin the task is as well in preDispatch(). You can launch an acl check for each requested url (so for each user request, even ajax). Here's a partial snippet, so this just an example of how you could handle things:

    public function preDispatch(Zend_Controller_Request_Abstract $request) {
        $controller = $request->controller;
        $module = $request->module;
        $action = $request->action;
        // here you should code something nice retrieving you Zend_Acl object
        // with some caching options maybe, building roles, ressources, etc
        $this->_acl = $this->getAcl(); 
        if (!$this->_acl->isCurrentUserAllowed($module,'see')) {
            $auth = Zend_Auth::getInstance();
        $identity  = $auth->hasIdentity('identity')? $auth->getIdentity() : null;
        if(isset($identity)) {
                if($identity['name'] == 'anonymous') {
                    // WARNING: avoid infinite redirect loops on login page
                    if (!($request->getControllerName() == 'login' 
                        && $request->getActionName()=='login' 
                        && $request->getModuleName() == 'default')) {
                            $request->setControllerName('login')
                   ->setActionName('login')
                   ->setModuleName('default');
                return;
    (...)
    

    and in this system the last important part is the LoginController where in case of succesful login you should initate the identity record:

    (...)
    $auth = Zend_Auth::getInstance();
    Zend_Session::regenerateId();
    $storage = $auth->getStorage();
    $rowobject = $authAdapter->getResultRowObject(null,'passwd');
    $storage->write((array)$rowobject);
    (...)
    

提交回复
热议问题