{"id":177567,"date":"2013-04-23T01:30:03","date_gmt":"2013-04-22T21:30:03","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=177567"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=177567","title":{"rendered":"<span class=\"post_title\">\u0421\u0435\u0440\u0432\u0435\u0440 \u043e\u0442\u0447\u0435\u0442\u043e\u0432 \u043d\u0430 django<\/span>"},"content":{"rendered":"<div class=\"content html_format\">   \t\u0414\u043e\u0431\u0440\u043e\u0433\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0441\u0443\u0442\u043e\u043a.<\/p>\n<p>  \u0422\u0430\u043a \u0441\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c, \u0447\u0442\u043e \u043c\u043e\u044f \u0440\u0430\u0431\u043e\u0442\u0430 \u0441\u0432\u044f\u0437\u0430\u043d\u0430 \u0441 \u043d\u0430\u043f\u0438\u0441\u0430\u043d\u0438\u0435\u043c \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<br \/>  \u042d\u0442\u043e\u043c\u0443 \u044f \u043f\u043e\u0441\u0432\u044f\u0442\u0438\u043b \u043e\u043a\u043e\u043b\u043e 8 \u043b\u0435\u0442. \u041e\u0442\u0447\u0435\u0442\u044b \u2014 \u044d\u0442\u043e \u0433\u043b\u0430\u0437\u0430 \u0431\u0438\u0437\u043d\u0435\u0441-\u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0430 \u0438 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f, <br \/>  \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u0430\u044f \u0434\u043b\u044f \u043f\u0440\u0438\u043d\u044f\u0442\u0438\u044f \u043e\u043f\u0435\u0440\u0430\u0442\u0438\u0432\u043d\u044b\u0445 \u0440\u0435\u0448\u0435\u043d\u0438\u0439.<\/p>\n<p>  \u0412\u043d\u0430\u0447\u0430\u043b\u0435 \u043d\u0430\u0448 \u043e\u0442\u0434\u0435\u043b \u0434\u0435\u043b\u0430\u043b \u043e\u0442\u0447\u0435\u0442\u044b, <br \/>   \u2014 \u041f\u0440\u0438\u043d\u0438\u043c\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0438 \u043f\u043e outlook<br \/>   \u2014 \u0421\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u044f sql-\u0437\u0430\u043f\u0440\u043e\u0441<br \/>   \u2014 \u041e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u044f \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b \u0437\u0430\u043a\u0430\u0437\u0447\u0438\u043a\u0443 \u0432 xls<br \/>   \u2014 \u0412 \u043b\u0443\u0447\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435, \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u044f sql-\u043a\u043e\u0434 \u043a\u0443\u0434\u0430-\u0442\u043e \u0432 \u043f\u0430\u043f\u043a\u0443 (\u0430 \u0438\u043d\u043e\u0433\u0434\u0430 \u0438 \u043d\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u044f)<\/p>\n<p>  \u041d\u043e \u044d\u0442\u043e \u0431\u044b\u043b\u043e \u0441\u043a\u0443\u0447\u043d\u043e \u0438 \u043d\u0435\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e. \u0422\u0430\u043a \u043f\u043e\u044f\u0432\u0438\u043b\u043e\u0441\u044c \u043f\u0440\u043e\u0441\u0442\u0435\u0439\u0448\u0435\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043d\u0430 PHP, <br \/>  \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043a\u0430\u0436\u0434\u044b\u0439 \u043e\u0442\u0447\u0435\u0442 \u0431\u044b\u043b \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u0435\u043d \u0432 \u0432\u0438\u0434\u0435 php-\u0444\u0430\u0439\u043b\u0430 \u0441 \u043e\u0434\u043d\u0438\u043c \u043a\u043b\u0430\u0441\u0441\u043e\u043c, \u0438\u043c\u0435\u044e\u0449\u0438\u043c \u0435\u0434\u0438\u043d\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 (\u043f\u043e\u043c\u0438\u043c\u043e \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0430) \u043c\u0435\u0442\u043e\u0434 show()<\/p>\n<p>  \u0412 \u0442\u0430\u043a\u043e\u043c \u0432\u0438\u0434\u0435, \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043f\u0440\u043e\u0436\u0438\u043b\u0430 5,5 \u043b\u0435\u0442, \u0437\u0430 \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043c\u043d\u043e\u0439 \u0438 \u0435\u0449\u0435 \u043e\u0434\u043d\u0438\u043c \u0447\u0435\u043b\u043e\u0432\u0435\u043a\u043e\u043c \u0431\u044b\u043b\u043e \u043d\u0430\u043f\u0438\u0441\u0430\u043d\u043e \u0431\u043e\u043b\u0435\u0435 500 \u0440\u0430\u0437\u043b\u0438\u0447\u043d\u044b\u0445 \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<br \/>  \u0412 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u043f\u043e\u044f\u0432\u0438\u043b\u0441\u044f \u043e\u043f\u044b\u0442 \u0438 \u0441\u0442\u0430\u043b\u043e \u043f\u043e\u043d\u044f\u0442\u043d\u043e, \u0447\u0442\u043e \u043c\u043d\u043e\u0433\u043e\u0435 (\u0435\u0441\u043b\u0438 \u043d\u0435 \u0432\u0441\u0435) \u0441\u0434\u0435\u043b\u0430\u043d\u043e \u043d\u0435 \u0442\u0430\u043a, \u0434\u0430 \u0438 PHP \u0443\u0436\u0435 \u043d\u0435 \u0443\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u043b.<\/p>\n<p>  \u0421\u0435\u0440\u0432\u0435\u0440 \u043e\u0442\u0447\u0435\u0442\u043e\u0432 \u0431\u044b\u043b \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u043d \u043d\u0430 django, \u0433\u0434\u0435 \u0431\u044b\u043b\u0430 \u00ab\u0430\u0434\u043c\u0438\u043d\u043a\u0430\u00bb \u0438 \u043a\u043e\u0434 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0443\u0436\u0435 \u043d\u0435 \u043f\u0440\u0430\u0432\u0438\u043b\u0441\u044f.<br \/>  \u0412 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0440\u0430\u0431\u043e\u0442\u044b \u0441\u043d\u043e\u0432\u0430 \u043d\u0430\u043a\u043e\u043f\u0438\u043b\u043e\u0441\u044c \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043c\u044b\u0441\u043b\u0435\u0439, <br \/>  \u0432 \u0438\u0442\u043e\u0433\u0435 \u0441\u0435\u0440\u0432\u0435\u0440 \u0431\u044b\u043b \u0441\u043d\u043e\u0432\u0430 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u043d.<\/p>\n<p>  \u042f \u0440\u0435\u0448\u0438\u043b \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0441\u043e\u043e\u0431\u0449\u0435\u0441\u0442\u0432\u0443 \u0442\u0435\u043a\u0443\u0449\u0443\u044e \u0432\u0435\u0440\u0441\u0438\u044e \u0441\u0438\u0441\u0442\u0435\u043c\u044b, \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e, \u043e\u043d\u0430 \u043a\u043e\u043c\u0443-\u0442\u043e \u043e\u0431\u043b\u0435\u0433\u0447\u0438\u0442 \u0436\u0438\u0437\u043d\u044c,<br \/>  \u0441\u043d\u044f\u0432 \u0440\u0443\u0442\u0438\u043d\u0443 \u043e\u0442\u0447\u0435\u0442\u043d\u043e\u0441\u0442\u0438 \u0438 \u043f\u0435\u0440\u0435\u043b\u043e\u0436\u0438\u0442 \u0435\u0435 \u043d\u0430 \u043c\u0430\u0448\u0438\u043d\u0443, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0433\u043e\u0442\u043e\u0432\u0430 \u0441\u043e\u0442\u043d\u0438 \u0440\u0430\u0437 \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043e\u0442\u0447\u0435\u0442\u044b<br \/>  \u0442\u044b\u0441\u044f\u0447\u0438 \u043b\u044e\u0434\u044f\u043c \u043f\u043e \u0441\u043e\u0442\u043d\u0438 \u0440\u0430\u0437\u043d\u044b\u0445 \u0432\u0445\u043e\u0434\u043d\u044b\u0445 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u0432.<\/p>\n<p>  \u042d\u0442\u043e \u0440\u0430\u0431\u043e\u0447\u0438\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u044b \u043c\u043e\u0436\u0435\u0442\u0435 (\u0435\u0441\u043b\u0438 \u0437\u0430\u0445\u043e\u0442\u0438\u0442\u0435) \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432 \u0440\u0430\u0431\u043e\u0442\u0435,<br \/>  \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043d\u0430\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u043f\u0440\u043e\u0441\u0442\u043e \u043d\u0440\u0430\u0432\u0438\u0442\u0441\u044f.<\/p>\n<p>  <a name=\"habracut\"><\/a><\/p>\n<h2>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/h2>\n<p>  \u0412\u0441\u0435 \u0437\u0430\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043e\u0432\u0430\u043d\u043d\u044b\u0435, \u043c\u043e\u0433\u0443\u0442 \u0441\u0440\u0430\u0437\u0443 \u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u0442\u044c\u0441\u044f \u0441 \u0438\u0441\u0445\u043e\u0434\u043d\u044b\u043c \u043a\u043e\u0434\u043e\u043c \u043d\u0430 <a href=\"https:\/\/bitbucket.org\/dibrovsd\/py_docflow\/\">https:\/\/bitbucket.org<\/a><\/p>\n<h2>\u041a\u043e\u043d\u0446\u0435\u043f\u0446\u0438\u044f \u0438 \u043e\u0441\u043d\u043e\u0432\u043d\u044b\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b<\/h2>\n<p>  \u0421\u0438\u0441\u0442\u0435\u043c\u0430 \u0441\u043e\u0441\u0442\u043e\u0438\u0442 \u0438\u0437 \u043d\u0430\u0431\u043e\u0440\u0430 \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<br \/>   \u2014 \u041e\u0442\u0447\u0435\u0442\u044b \u0441\u043b\u0430\u0431\u043e (\u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043d\u0438\u043a\u0430\u043a) \u043d\u0435 \u0441\u0432\u044f\u0437\u0430\u043d\u044b \u0434\u0440\u0443\u0433 \u0441 \u0434\u0440\u0443\u0433\u043e\u043c, \u0447\u0442\u043e\u0431 \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u043c\u043e\u0433 \u0431\u044b\u0442\u044c \u0443\u0432\u0435\u0440\u0435\u043d, \u0447\u0442\u043e \u043d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u0438\u0441\u043f\u043e\u0440\u0442\u0438\u0442 \u0432 \u0434\u0440\u0443\u0433\u0438\u0445 \u043e\u0442\u0447\u0435\u0442\u0430\u0445, \u043e\u0442\u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0432 \u043a\u0430\u043a\u043e\u0439-\u0442\u043e \u043e\u0434\u0438\u043d.<br \/>   \u2014 \u042d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043e\u0442\u0447\u0435\u0442\u0430 \u0436\u0435\u0441\u0442\u043a\u043e \u0441\u0432\u044f\u0437\u0430\u043d\u043d\u044b \u0432\u043d\u0443\u0442\u0440\u0438 \u043e\u0442\u0447\u0435\u0442\u0430 \u0438 \u0432\u043b\u0438\u044f\u044e\u0442 \u0434\u0440\u0443\u0433 \u043d\u0430 \u0434\u0440\u0443\u0433\u0430.<\/p>\n<p>  \u041e\u0442\u0447\u0435\u0442\u044b \u043d\u0435 \u043e\u0440\u0433\u0430\u043d\u0438\u0437\u043e\u0432\u0430\u043d\u044b \u0432 \u0438\u0435\u0440\u0430\u0440\u0445\u0438\u044e (\u043e\u0442 \u044d\u0442\u043e\u0433\u043e \u0443\u0448\u043b\u0438), \u0430 \u043f\u043e\u043c\u0435\u0447\u0430\u044e\u0442\u0441\u044f \u043d\u0430\u0431\u043e\u0440\u043e\u043c \u0442\u044d\u0433\u043e\u0432.<br \/>  \u041d\u0430 \u0433\u043b\u0430\u0432\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u044b \u043e\u0442\u0447\u0435\u0442\u043e\u0432 \u043c\u043e\u0436\u043d\u043e \u0449\u0435\u043b\u0447\u043a\u0430\u043c\u0438 \u043d\u0430\u0431\u0438\u0440\u0430\u0442\u044c \u043d\u0443\u0436\u043d\u044b\u0435 \u0441\u043e\u0447\u0435\u0442\u0430\u043d\u0438\u044f \u0442\u044d\u0433\u043e\u0432, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0442, \u043a\u0430\u043a \u0444\u0438\u043b\u044c\u0442<br \/>  \u0438 \u043f\u043e\u0438\u0441\u043a \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<\/p>\n<p>  \u042d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043e\u0442\u0447\u0435\u0442\u0430 \u00ab\u043b\u0435\u043d\u0438\u0432\u044b\u0435\u00bb \u0438 \u043d\u0430\u0447\u0438\u043d\u0430\u044e\u0442 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0442\u0441\u044f \u0432 \u043c\u043e\u043c\u0435\u043d\u0442 \u0441\u0431\u043e\u0440\u043a\u0438 \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u043e\u0432\u0449\u0438\u043a\u0430 \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432,<br \/>  \u0447\u0442\u043e \u0434\u0430\u0435\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0442\u0435 \u0437\u0430\u043f\u0440\u043e\u0441\u044b \u043a \u0431\u0430\u0437\u0435, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b,<br \/>  \u0440\u0430\u0437\u043c\u0435\u0449\u0430\u044f \u0432 \u043e\u0442\u0447\u0435\u0442\u0435 \u0438 \u0433\u0440\u0443\u043f\u043f\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438 \u0434\u0435\u0442\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e.<br \/>  \u0417\u0430 \u0441\u0447\u0435\u0442 \u043a\u044d\u0448\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0434\u0430\u043d\u043d\u044b\u0445 \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430, \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432, \u0432\u044b\u0432\u043e\u0434\u044f\u0449\u0438\u0445 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438\u0437 \u043e\u0434\u043d\u043e\u0433\u043e \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430, \u0434\u0430\u044e\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043e\u0434\u0438\u043d \u0437\u0430\u043f\u0440\u043e\u0441 \u0432 \u0431\u0430\u0437\u0443 \u0434\u0430\u043d\u043d\u044b\u0445.  <\/p>\n<pre><code class=\"html\">&lt;!--  \u0412\u044b\u043f\u043e\u043b\u043d\u044f\u044e\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e \u0442\u0435 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u0434\u0430\u043d\u043d\u044b\u0445, \u043d\u0430 \u043a\u043e\u0442\u043e\u0440\u044b\u0445 \u043e\u0441\u043d\u043e\u0432\u0430\u043d\u044b \u0432\u0438\u0434\u0436\u0435\u0442\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u0430\u043d\u044b --&gt; {% if get.detail == 'table' %}     {{table.some_table_detail}} {% elif get.detail == 'chart' %}     {{charts.some_chart}} {% else %}         {{tables.grouped_table}} {% endif %} <\/code><\/pre>\n<p>  \u0415\u0441\u0442\u044c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e \u043a \u043e\u0442\u0447\u0435\u0442\u0443, \u043f\u043e\u043c\u0438\u043c\u043e \u0441\u0430\u043c\u043e\u0433\u043e \u0444\u0430\u043a\u0442\u0430 \u0434\u043e\u0441\u0442\u0443\u043f\u043d\u043e\u0441\u0442\u0438.<br \/>  \u041a\u043e\u0442\u043e\u0440\u044b\u0435 \u043c\u043e\u0433\u0443\u0442 \u0431\u044b\u0442\u044c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u044b \u0434\u043b\u044f \u043f\u0440\u0435\u0434\u043e\u0441\u0442\u0430\u0432\u043b\u0435\u043d\u0438\u0438 \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043a \u0447\u0430\u0441\u0442\u0438 \u043e\u0442\u0447\u0435\u0442\u0430 \u0438\u043b\u0438 \u0447\u0430\u0441\u0442\u0438 \u0434\u0430\u043d\u043d\u044b\u0445  <\/p>\n<pre><code class=\"sql\">select * from some_table t where 1 = 1 -- \u042d\u0442\u043e \u0438\u043d\u044a\u0435\u043a\u0446\u0438\u044f {% if user_params.only_filials %}and filial_id in ({{user_params.only_filials|join:&quot;,&quot;}}){% endif %} -- \u042d\u0442\u043e \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u043d\u043d\u0430\u044f \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u0430\u044f, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0437\u0430\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f \u043d\u0430 %s  {% if user_params.only_sectors %}and sector_id = [[user_params.only_sectors.0]]{% endif %} <\/code><\/pre>\n<p>  \u041f\u0440\u0438 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0434\u043d\u043e\u0433\u043e \u0432\u0430\u0440\u0438\u0430\u043d\u0442\u0430 \u0432\u044b\u0437\u043e\u0432\u0430 \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u0430 \u0432 \u0448\u0430\u0431\u043b\u043e\u043d\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u0435, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f __call__<br \/>  \u0415\u0441\u043b\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u044b \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0432\u0430\u0440\u0438\u0430\u043d\u0442\u043e\u0432, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f __getitem__ \u0434\u043b\u044f \u0441\u043b\u043e\u0432\u0430\u0440\u043d\u043e\u0433\u043e \u043e\u0431\u044a\u0435\u043a\u0442\u0430.<\/p>\n<p>  \u041a\u0430\u0436\u0434\u044b\u0439 \u043e\u0442\u0447\u0435\u0442 \u043c\u043e\u0436\u0435\u0442 \u0441\u043e\u0441\u0442\u043e\u044f\u0442\u044c \u0438\u0437:<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u041c\u0435\u043d\u0435\u0434\u0436\u0435\u0440 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class EnvirementManager(object):     u'''     \u0420\u0430\u0431\u043e\u0442\u0430 \u0441 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c\u0438 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f, render \u0441\u0442\u0440\u043e\u043a \u0432\u043d\u0443\u0442\u0440\u0438 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f,     \u0440\u0430\u0437\u0431\u043e\u0440 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0438 \u0442.\u043f.     '''      def __init__(self, **kwargs):          self._env = kwargs      def render(self, text, **dict):         u''' \u041e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0430 \u0441\u0442\u0440\u043e\u043a\u0438 \u0448\u0430\u0431\u043b\u043e\u043d\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u043e\u0439 '''          return render_string(text, self._dict_mix(dict))          def render_template_compiled(self, template, **dict):         u''' \u041e\u0431\u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u043f\u0440\u0435\u0434\u0432\u0430\u0440\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0441\u043a\u043e\u043c\u043f\u0438\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0448\u0430\u0431\u043b\u043e\u043d '''                  return template.render(Context(self._dict_mix(dict)))          def render_template(self, path, **dict):         u''' \u041e\u0431\u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c '''                  return render_to_string(path,                                  self._dict_mix(dict),                                  context_instance=RequestContext(self._env['request']))          def render_sql(self, sql, **dict):         u'''  \u041e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0435\u0442 \u0441\u0442\u0440\u043e\u043a\u0443 \u0448\u0430\u0431\u043b\u043e\u043d\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u043e\u0439         \u0412\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0430\u043d\u043d\u0443\u044e \u0441\u0442\u0440\u043e\u043a\u0443 \u0438 \u043c\u0430\u0441\u0441\u0438\u0432 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0434\u043b\u044f \u043f\u0440\u0438\u0432\u044f\u0437\u043a\u0438 '''          sql_rendered = self.render(sql, **dict)         binds = []          for bind_token in re.findall(r'\\[{2}.+\\]{2}', sql_rendered):             env_path = bind_token[2:-2]             binds.append(attribute_by_name(self._env, env_path))             sql_rendered = sql_rendered.replace(bind_token, u'%s')          return (sql_rendered, binds)          def _dict_mix(self, dict):                  if dict:             e = self._env.copy()             e.update(dict)         else:             e = self._env                      return e      def get(self, name, value=None):                  return self._env.get(name, value)          def __getitem__(self, name):                  return self._env[name]      def __setitem__(self, name, value):                  self._env[name] = value <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0418\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0438\u043c\u0435\u043d\u043e\u0432\u0430\u043d\u043d\u044b\u043c\u0438 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438 \u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u0434\u043b\u044f \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430 \u0432\u0441\u0435\u0433\u043e <br \/>  \u0441 \u0443\u0447\u0435\u0442\u043e\u043c \u0442\u0435\u0445 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445, \u043a\u043e\u0442\u043e\u0440\u044b\u043c\u0438 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u043d.<\/p>\n<p>  \u0417\u0430 \u0441\u0447\u0435\u0442 \u044d\u0442\u043e\u0433\u043e, \u043c\u044b \u043c\u043e\u0436\u0435\u043c \u0433\u0430\u0440\u0430\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u0438\u0437 \u043b\u044e\u0431\u043e\u0433\u043e \u043c\u0435\u0441\u0442\u0430 \u0437\u0430\u0440\u0430\u043d\u0435\u0435 \u0438\u0437\u0432\u0435\u0441\u0442\u043d\u043e\u0433\u043e \u043d\u0430\u0431\u043e\u0440\u0430 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445.<br \/>  \u041c\u043e\u0436\u0435\u0442 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u0442\u044c sql-\u0437\u0430\u043f\u0440\u043e\u0441\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u044f\u0432\u043b\u044f\u044e\u0442\u0441\u044f \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430\u043c\u0438 \u0434\u0430\u043d\u043d\u044b\u0445 (\u043d\u043e \u043e\u0431 \u044d\u0442\u043e\u043c \u0447\u0443\u0442\u044c \u043d\u0438\u0436\u0435)<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u041c\u0435\u043d\u0435\u0434\u0436\u0435\u0440 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u043e\u0432 \u0434\u0430\u043d\u043d\u044b\u0445<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class DatasetManager(object):     u''' \u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430\u043c\u0438 \u0434\u0430\u043d\u043d\u044b\u0445 '''      def __init__(self, request, dataset, env):         u'''         \u041a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440         '''          self._request = request         self._dataset = dataset         self._env = env         self._cache = []         self._xml = None          if self._dataset.xml_settings:             self._xml = etree.fromstring(self._env.render(self._dataset.xml_settings))      def get_data(self):         u'''         \u0412\u044b\u043f\u043e\u043b\u043d\u044f\u0435\u0442 \u0437\u0430\u043f\u0440\u043e\u0441 \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0441\u043b\u043e\u0432\u0430\u0440\u044c \u0441 \u0434\u0430\u043d\u043d\u044b\u043c\u0438         '''          if not self._cache:              self._cache = [[], []]             (sql, binds) = self._env.render_sql(self._dataset.sql)             cursor = self._modify_cursor(connections['reports'].cursor())             cursor.execute(sql, binds)              # \u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u043a\u043e\u043b\u043e\u043d\u043e\u043a \u0437\u0430\u043f\u0440\u043e\u0441\u0430             xml_columns = {}             if self._xml not in (None, ''):                 for xml_column in self._xml.xpath('\/xml\/columns\/column'):                     attrs = xml_column.attrib                     xml_columns[force_unicode(attrs['name'])] = force_unicode(attrs['alias'])              # \u041a\u043e\u043b\u043e\u043d\u043a\u0438 \u0437\u0430\u043f\u0440\u043e\u0441\u0430 (\u0437\u0430\u043c\u0435\u043d\u044f\u0435\u043c \u043d\u0430\u0437\u0432\u0430\u043d\u0438\u0435 \u043a\u043e\u043b\u043e\u043d\u043a\u0438, \u0435\u0441\u043b\u0438 \u0437\u0430\u043f\u0440\u043e\u0448\u0435\u043d\u043e)             # \u044d\u0442\u043e \u043c\u043e\u0436\u0435\u0442 \u043f\u0440\u0438\u0433\u043e\u0434\u0438\u0442\u0441\u044f \u043f\u0440\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0438 \u0411\u0414, \u0437\u0430\u043f\u0440\u0435\u0449\u0430\u044e\u0435\u0439 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0434\u043b\u0438\u043d\u043d\u044b\u0445 \u0438\u043c\u0435\u043d             for field in cursor.description:                 name_unicode = force_unicode(field.name)                 self._cache[0].append(xml_columns[name_unicode] if name_unicode in xml_columns                                                                 else name_unicode)              self._cache[1] = cursor.fetchall()          return self._cache      def __getitem__(self, name):         u''' \u0432\u044b\u0437\u043e\u0432\u044b \u0438\u0437 \u0448\u0430\u0431\u043b\u043e\u043d\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u044b \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0442 \u0447\u0435\u0440\u0435\u0437 \u0441\u043b\u043e\u0432\u0430\u0440\u043d\u044b\u0435 \u043e\u0431\u044a\u0435\u043a\u0442\u044b '''          if name == 'sql':             return self._env.render(self._dataset.sql)          elif name == 'render':             (sql, binds) = self._env.render_sql(self._dataset.sql)             return {'sql': sql, 'binds': binds}          elif name == 'data':             (fields, data) = self.get_data()             return [dict(zip(fields, row)) for row in data]      def _modify_cursor(self, cursor):         u'''         \u041c\u043e\u0434\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u044f \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u0432 \u043a\u0443\u0440\u0441\u043e\u0440\u0430 (\u043c\u043e\u0436\u0435\u0442 \u043f\u043e\u0442\u0440\u0435\u0434\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0434\u043b\u044f \u0440\u0430\u0437\u043d\u044b\u0445 \u0431\u0430\u0437 \u0431\u0430\u0437 \u0434\u0430\u043d\u043d\u044b\u0445),         \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0434\u043b\u044f Oracle \u0442\u0440\u0435\u0431\u0443\u0435\u0442\u0441\u044f         \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u0432 \u043b\u043e\u043a\u0430\u043b\u0438, \u043e\u0442\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 number_as_string = True (\u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e \u0432 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u043e\u043c         backends django)         '''          return cursor  <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u041f\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u0435\u0442:<br \/>   \u2014 \u0414\u0430\u043d\u043d\u044b\u0435 \u0432 \u0432\u0438\u0434\u0435 \u043a\u043e\u0440\u0442\u0435\u0436\u0430 (field_names, rows,)<br \/>   \u0414\u043b\u044f \u0432\u0438\u0437\u0443\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0440\u0430\u0437\u043b\u0438\u0447\u043d\u044b\u043c\u0438 \u0442\u0438\u043f\u0430\u043c\u0438 \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432, \u0432\u044b\u0433\u0440\u0443\u0437\u043a\u0438 \u0432 excel<br \/>   \u2014 \u0414\u0430\u043d\u043d\u044b\u0435 \u0432 \u0432\u0438\u0434\u0435 \u0441\u043f\u0438\u0441\u043a\u0430 \u0441\u043b\u043e\u0432\u0430\u0440\u0435\u0439 (\u043a\u0430\u0436\u0434\u0430\u044f \u0441\u0442\u0440\u043e\u043a\u0430 \u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0441\u043b\u043e\u0432\u0430\u0440\u0435\u043c)<br \/>   \u0414\u043b\u044f \u043f\u0440\u044f\u043c\u043e\u0433\u043e \u043e\u0431\u0440\u0430\u0449\u0435\u043d\u0438\u044f \u043a \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0443 \u0438 \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 html, javascript, css \u043a\u043e\u0434\u0430 \u0432 \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u043e\u0432\u043a\u0438 (\u043e \u043d\u0435\u043c \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u043e\u0437\u0436\u0435) <br \/>   \u2014 sql-\u043a\u043e\u0434 \u0431\u0435\u0437 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0438 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u043f\u0440\u0438\u0432\u044f\u0437\u043a\u0438<br \/>   \u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u0434\u043b\u044f \u043e\u0440\u0433\u0430\u043d\u0438\u0437\u0430\u0446\u0438\u0438 \u0432\u043b\u043e\u0436\u0435\u043d\u043d\u044b\u0445 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u043e\u0432 \u0434\u0430\u043d\u043d\u044b\u0445, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440:   <\/p>\n<pre><code class=\"sql\">  select key, count(1) from ({{datasets.base_dataset.sql}}) group by key   <\/code><\/pre>\n<p>   \u2014 sql-\u043a\u043e\u0434 \u0438 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043f\u0440\u0438\u0432\u044f\u0437\u043a\u0438 <br \/>   \u041f\u0440\u043e\u0441\u0442\u043e \u0434\u043b\u044f \u043e\u0442\u043b\u0430\u0434\u043a\u0438 \u0438 \u0432\u044b\u0432\u043e\u0434\u0435 \u0432 \u043e\u0442\u0447\u0435\u0442<\/p>\n<p>  \u0417\u0430 \u0441\u0447\u0435\u0442 \u044d\u0442\u043e\u0433\u043e, \u043c\u043e\u0436\u043d\u043e:<br \/>   \u2014 \u041c\u0435\u043d\u044f\u0442\u044c \u043d\u0430 \u043b\u0435\u0442\u0443 sql-\u0437\u0430\u043f\u0440\u043e\u0441 \u0432 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u0430 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f<br \/>   \u2014 \u041d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u043e\u0434\u043d\u043e\u0433\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u0430, \u043c\u043e\u0436\u043d\u043e \u0434\u0435\u043b\u0430\u0442\u044c \u0434\u0440\u0443\u0433\u0438\u0435 \u0442\u0430\u043a, \u0447\u0442\u043e\u0431 \u0443 \u0432\u0430\u0441 \u0433\u0440\u0443\u043f\u043f\u0438\u0440\u043e\u0432\u043a\u0430 \u0438 \u0434\u0435\u0442\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u043d\u0438\u043a\u043e\u0433\u0434\u0430 \u043d\u0435 \u0431\u0443\u0434\u0443\u0442 \u043e\u0442\u043b\u0438\u0447\u0430\u0442\u0441\u044f,<br \/>  \u0442\u0430\u043a \u043a\u0430\u043a \u043e\u0441\u043d\u043e\u0432\u0430\u043d\u044b \u043d\u0430 \u043e\u0434\u043d\u043e\u043c \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0435, \u0438 \u0432\u044b \u043d\u0438\u043a\u043e\u0433\u0434\u0430 \u043d\u0435 \u0437\u0430\u0431\u0443\u0434\u0438\u0442\u0435 \u0434\u043e\u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0433\u0440\u0443\u043f\u043f\u0438\u0440\u043e\u0432\u043a\u0443 \u0438 \u0437\u0430\u0431\u044b\u0442\u044c \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u044d\u0442\u043e \u0432 \u0434\u0435\u0442\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438<br \/>   \u2014 \u0412\u044b\u0431\u043e\u0440\u043a\u0443 \u0438\u0437 \u043e\u0434\u043d\u043e\u0433\u043e \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430 \u0434\u0430\u043d\u043d\u044b\u0445 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0434\u043b\u044f \u0434\u0438\u043d\u0430\u043c\u0438\u0447\u0435\u0441\u043a\u043e\u0439 \u0441\u0431\u043e\u0440\u043a\u0438 \u0434\u0440\u0443\u0433\u043e\u0433\u043e.<br \/>  \u041d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0441\u043e\u0437\u0434\u0430\u0442\u044c \u0437\u0430\u0440\u0430\u043d\u0435\u0435 \u043d\u0435\u0438\u0437\u0432\u0435\u0441\u0442\u043d\u043e\u0435 \u043a\u043e\u043b-\u0432\u043e \u043a\u043e\u043b\u043e\u043d\u043e\u043a \u0432 \u0437\u0430\u043f\u0440\u043e\u0441\u0435 \u043f\u043e\u043c\u0435\u0441\u044f\u0447\u043d\u043e, \u043f\u043e \u043c\u0435\u0441\u044f\u0446\u0430\u043c (\u0438\u043b\u0438 \u043d\u0435\u0434\u0435\u043b\u044f\u043c), \u0432\u0445\u043e\u0434\u044f\u0449\u0438\u043c \u0432 \u043e\u0442\u0447\u0435\u0442\u043d\u044b\u0439 \u043f\u0435\u0440\u0438\u043e\u0434 (\u0437\u0430\u0434\u0430\u0435\u0442\u0441\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u043c \u0432 \u0444\u043e\u0440\u043c\u0435, \u043d\u043e \u043e \u043d\u0438\u0445 \u0442\u043e\u0436\u0435 \u0447\u0443\u0442\u044c \u043d\u0438\u0436\u0435),<br \/>  \u0438 \u0432\u0441\u0435 \u044d\u0442\u043e \u043d\u0435\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e \u043e\u0442 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445 (\u0431\u0435\u0437 pivot|unpivot oracle \u0438 xml_query oracle)<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u041c\u0435\u043d\u0435\u0434\u0436\u0435\u0440 \u0444\u0438\u043b\u044c\u0442\u0440\u043e\u0432<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class FilterManager(object):     u''' \u041a\u043b\u0430\u0441\u0441 \u0434\u043b\u044f \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u043c\u0438 '''          def __init__(self, request, filter_obj, env):         u''' \u041a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 '''                  self._request = request         self._filter_obj = filter_obj         self._env = env         self._form_instance = None         self._changed_data = {}                  # \u0415\u0441\u043b\u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u043b\u0438 \u044d\u0442\u0443 \u0444\u043e\u0440\u043c\u0443, \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0444\u043e\u0440\u043c\u0443 POST-\u0434\u0430\u043d\u043d\u044b\u043c\u0438         # \u041f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438, \u0435\u0441\u043b\u0438 \u043e\u043d\u0438 \u043e\u0442\u043b\u0438\u0447\u0430\u044e\u0442\u0441\u044f \u043e\u0442 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u043d\u044b\u0445, \u0437\u0430\u043f\u043e\u043c\u043d\u0438\u043c \u0438\u0445 \u0432 self._changed_data         if self._request.method == 'POST' and int(self._request.POST['filter_id']) == self._filter_obj.id:             self._form_instance = self._get_form_class()(self._request.POST)             if self._form_instance.is_valid():                 data = self._form_instance.cleaned_data                 for key, value in data.items():                     if self._env['f'].get(key) != value:                         self._changed_data[key] = value              def get_changed_data(self):                  return self._changed_data          def get_form(self):         u''' \u041f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u044d\u043a\u0437\u0435\u043c\u043f\u043b\u044f\u0440 \u0444\u043e\u0440\u043c\u044b \u0444\u0438\u043b\u044c\u0442\u0440\u0430 '''                  try:              # \u0415\u0441\u043b\u0438 \u044d\u043a\u0437\u0435\u043c\u043f\u043b\u044f\u0440 \u0444\u043e\u0440\u043c\u044b \u043d\u0435 \u0441\u043e\u0437\u0434\u0430\u043d \u0432 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0435, (\u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f\u0445)             # \u0437\u043d\u0430\u0447\u0438\u0442 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u043c \u0438\u0437 \u0437\u0430\u0440\u0430\u043d\u0435\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u043d\u044b\u0445 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u0439             if self._form_instance is None:                 self._form_instance = self._get_form_class()(initial=self._env['f'])                              html = self._env.render('{% load custom_filters %}' + self._filter_obj.html_layout, form=self._form_instance)             return self._env.render_template('reports_filter.html', form_html=html, filter=self._filter_obj)          except Exception as e:             return e          def __call__(self):         u''' \u0414\u043b\u044f \u0432\u044b\u0437\u043e\u0432\u0430 \u0432 \u0448\u0430\u0431\u043b\u043e\u043d\u0435 '''         try:             return self.get_form()         except Exception as e:             return e          def _get_form_class(self):         u''' \u0421\u043e\u0431\u0440\u0430\u0442\u044c \u0444\u043e\u0440\u043c\u0443 \u0444\u0438\u043b\u044c\u0442\u0440\u0430 \u0438 \u0432\u0435\u0440\u043d\u0443\u0442\u044c \u0435\u0435 '''                  form_attrs = {}                  filter_widgets = (DateField, ChoiceField, MultipleChoiceField, BooleanField, CharField, CharField, DateRangeField)              for item in self._filter_obj.form_items.all():                  kwargs = {'label': item.title, 'required': False}             if item.xml_settings:                 xml = xml = etree.fromstring(self._env.render(item.xml_settings))             else:                 xml = None                  if item.widget_type == 0:                 kwargs['widget'] = forms.DateInput(attrs={'class': 'date'})                  elif item.widget_type in (1, 2):                 choices = []                 for option in xml.xpath('\/xml\/options\/option'):                     choices.append((option.attrib['id'], option.attrib['value']))                 sql = xml.xpath('\/xml\/sql')                 if sql:                     curs = connections['reports'].cursor().execute(sql[0].text)                     for row in curs.fetchall():                         choices.append((row[0], row[1]))                 kwargs['choices'] = choices                  elif item.widget_type == 4:                 kwargs['max_length'] = 50                  elif item.widget_type == 5:                 default = xml.xpath('\/xml\/value')                 kwargs['widget'] = forms.HiddenInput(attrs={'value': default[0].text})                  form_attrs[item.key] = filter_widgets[item.widget_type](**kwargs)              filter_form = type(str(self._filter_obj.title), (forms.Form,), form_attrs)          return filter_form <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0422\u0443\u0442 \u0432\u0441\u0435 \u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e \u0442\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u043e. \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0444\u043e\u0440\u043c\u0443 \u0438 \u043e\u0442\u0434\u0430\u0435\u043c \u0444\u043e\u0440\u043c\u0443 \u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0422\u0430\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u0432\u0438\u0434\u0436\u0435\u0442<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class WidgetTableManager(object):     u''' \u0423\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u043e\u0439 \u0441 \u0442\u0430\u0431\u043b\u0438\u0447\u043d\u044b\u043c \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u043c '''      def __init__(self, request, widget_obj, dataset, env):         u''' \u041a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 '''          self._request = request         self._widget_obj = widget_obj         self._dataset = dataset         self._env = env         self._xml = None          if widget_obj.xml_settings:             self._xml = etree.fromstring(self._env.render(widget_obj.xml_settings).replace('[[', '{{').replace(']]', '}}'))      def get_html(self):         u''' \u0412\u0435\u0440\u043d\u0443\u0442\u044c html-\u043a\u043e\u0434 \u0442\u0430\u0431\u043b\u0438\u0446\u044b '''          (fields, data) = self._dataset.get_data()          field_settings = {}         table_settings = {}         if self._xml is not None:              table_settings_node = xml_node(self._xml, '\/xml\/table')             if table_settings_node is not None:                 table_settings = table_settings_node.attrib             for xml in self._xml.xpath('\/xml\/fields\/field'):                 xml_attributes = dict(xml.attrib)                 field_name = xml_attributes['name']                  if 'lnk' in xml_attributes:                     xml_attributes['tpl_lnk'] = Template(force_unicode(xml_attributes['lnk']))                 if 'cell_attributes' in xml_attributes:                     xml_attributes['tpl_cell_attributes'] = Template(force_unicode(xml_attributes['cell_attributes']))                  field_settings[field_name] = xml_attributes          # \u0412\u044b\u0432\u043e\u0434\u0438\u043c\u044b\u0435 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d \u043a\u043e\u043b\u043e\u043d\u043a\u0438         fields_visible = []         for index, field_name in enumerate(fields):             settings = field_settings.get(field_name, {})             if 'display' in settings and settings['display'] == '0':                 continue             fields_visible.append((index, field_name, settings))          # \u0412\u044b\u0447\u0438\u0441\u043b\u0438\u0442\u044c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0438 \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u0442\u044c \u043a \u0434\u0430\u043d\u043d\u044b\u043c         rows = []         for row in data:             row_dict = dict(zip(fields, row))             row_settings = {}              # \u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0443\u0440\u043e\u0432\u043d\u044f \u0441\u0442\u0440\u043e\u043a\u0438             if 'field_row_style' in table_settings:                 row_settings['row_style'] = row_dict[table_settings['field_row_style']]             if 'field_row_attributes' in table_settings:                 row_settings['row_attributes'] = row_dict[table_settings['field_row_attributes']]              # \u041f\u0435\u0440\u0435\u0431\u0440\u0430\u0442\u044c \u0441\u0442\u0440\u043e\u043a\u0438 \u0438 \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0443\u0440\u043e\u0432\u043d\u044f \u0441\u0442\u0440\u043e\u043a\u0438             fields_set = []             for index, field_name, settings in fields_visible:                 field = {'name': field_name, 'value': row[index]}                  # \u0415\u0441\u043b\u0438 \u0435\u0441\u0442\u044c \u0441\u0441\u044b\u043b\u043a\u0430 \u0438 \u043e\u043d\u0430 \u0434\u043e\u043b\u0436\u043d\u0430 \u0431\u044b\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u0430\u043d\u0430                 if 'tpl_lnk' in settings and ('lnk_enable_field' not in settings or row_dict[settings['lnk_enable_field']] not in (0, '0', '', None)):                     field['lnk'] = self._env.render_template_compiled(settings['tpl_lnk'], row=row_dict)                  # \u0410\u0442\u0442\u0440\u0438\u0431\u0443\u0442\u044b \u044f\u0447\u0435\u0439\u043a\u0438                 if 'tpl_cell_attributes' in settings:                     field['cell_attributes'] = settings['tpl_cell_attributes'].render(Context(row_dict))                  field['settings'] = settings                 fields_set.append(field)              rows.append({'settings': row_settings, 'fields': fields_set})          return render_to_string('reports_widget_table.html', {'fields': fields_visible, 'rows': rows, 'widget_obj': self._widget_obj})      def __call__(self):         u''' \u041f\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0435, \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0442\u0430\u0431\u043b\u0438\u0446\u0443 '''          try:             return self.get_html()         except Exception as e:             return u'\u041e\u0448\u0438\u0431\u043a\u0430 \u0432 \u0432\u0438\u0434\u0436\u0435\u0442\u0435 %s: &quot;%s&quot;' % (self._widget_obj.title, e)  <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0411\u0435\u0440\u0435\u0442 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438\u0437 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430 \u0434\u0430\u043d\u043d\u044b\u0445 \u0438 \u0432\u044b\u0432\u043e\u0434\u0438\u0442 \u0432 \u0442\u0430\u0431\u043b\u0438\u0447\u043d\u043e\u043c \u0432\u0438\u0434\u0435. \u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 <br \/>   \u2014 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435<br \/>   \u2014 \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e \u0441\u0441\u044b\u043b\u043e\u043a (\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043c\u043e\u0433\u0443\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0434\u043b\u044f \u0434\u0435\u0442\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u044f\u0447\u0435\u0439\u043a\u0438 \u0432 \u0434\u0440\u0443\u0433\u043e\u0439 \u0442\u0430\u0431\u043b\u0438\u0446\u0435, \u0433\u0440\u0430\u0444\u0438\u043a \u0438\u043b\u0438 excel)<br \/>   \u2014 \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u043b\u044c\u043d\u044b\u0445 \u0430\u0442\u0442\u0440\u0438\u0431\u0443\u0442\u043e\u0432 \u0441\u0442\u0440\u043e\u043a (\u0434\u043b\u044f \u0440\u0430\u0431\u043e\u0442\u044b javascript, \u0441\u043a\u0440\u044b\u0442\u0438\u044f \u0438\u043b\u0438 \u043f\u043e\u043a\u0430\u0437\u0430 \u0438\u0442\u043e\u0433\u043e\u0432\u044b\u0445 \u0441\u0442\u0440\u043e\u043a \u0438 \u0442.\u043f.)<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0412\u0438\u0434\u0436\u0435\u0442 &#8212; \u0433\u0440\u0430\u0444\u0438\u043a<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class WidgetChartManager(object):     u''' \u041c\u0435\u043d\u0435\u0434\u0436\u0435\u0440 \u0433\u0440\u0430\u0444\u0438\u043a\u0430 '''      def __init__(self, request, chart_obj, dataset, env):         u''' \u041a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 '''          self._request = request         self._chart_obj = chart_obj         self._dataset = dataset         self._env = env         self._xml = etree.fromstring(self._env.render(self._chart_obj.xml_settings))          print 1      def __call__(self):         u''' \u041f\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0435 \u0438\u0437 \u0448\u0430\u0431\u043b\u043e\u043d\u0430 '''          print 2         try:             return self.get_chart()         except Exception as e:             print unicode(e)             return unicode(e)      def get_chart(self):         u''' html \u0441\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u043c \u0441\u0431\u043e\u0440\u043a\u0438 \u0433\u0440\u0430\u0444\u0438\u043a\u0430 '''          (fields, data) = self._dataset.get_data()         return self._env.render_template('reports_widget_chart.html',                                          settings=xml_to_dict(xml_node(self._xml, '\/xml')),                                          data=json.dumps([dict(zip(fields, row)) for row in data], cls=JSONEncoder),                                          chart_obj=self._chart_obj,                                          )  <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0422\u0443\u0442 \u0442\u043e\u0436\u0435 \u043f\u0440\u043e\u0441\u0442\u043e. \u0411\u0435\u0440\u0435\u0442 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438\u0437 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430, XML-\u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u043f\u0435\u0440\u0435\u0432\u043e\u0434\u0438\u0442 \u0432 dict \u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u0442 \u0448\u0430\u0431\u043b\u043e\u043d,<br \/>  \u0441\u043e\u0431\u0438\u0440\u0430\u044f javascript-\u043a\u043e\u0434 \u0433\u0440\u0430\u0444\u0438\u043a\u0430 (\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f amcharts)<br \/>  \u0442\u044d\u0433\u0438 XML-\u0443\u0437\u043b\u043e\u0432 \u043f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u0443\u044e\u0442\u0441\u044f \u0432 \u043d\u0430\u0437\u0432\u0430\u043d\u0438\u0435 \u043f\u0430\u0440\u0430\u043c\u0435\u0440\u0442\u0430, \u0442\u0435\u043a\u0441\u0442 \u0432 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u0430,<br \/>  \u0442\u043e \u0435\u0441\u0442\u044c \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u0432\u0441\u0435 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438 amcharts, <br \/>  \u043f\u0440\u043e\u0441\u0442\u043e \u043f\u043e\u043c\u0435\u0441\u0442\u0438\u0432 \u043d\u0443\u0436\u043d\u044b\u0439 \u0442\u044d\u0433 \u0443 \u043d\u0443\u0436\u043d\u0443\u044e \u0441\u0435\u043a\u0446\u0438\u044e<\/p>\n<p>  \u0418, \u043a\u0430\u043a \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u0435 \u0442\u0435\u043e\u0440\u0435\u0442\u0438\u0447\u0435\u0441\u043a\u043e\u0439 \u0447\u0430\u0441\u0442\u0438, \u043f\u0440\u0438\u0432\u043e\u0436\u0443 \u043a\u043e\u0434 \u043a\u043b\u0430\u0441\u0441\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u0441\u0435\u043c \u044d\u0442\u0438\u043c \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442,<br \/>  \u0440\u0430\u0437\u043c\u0435\u0449\u0430\u044f \u0432\u0438\u0434\u0436\u0435\u0442\u044b, \u0438\u043b\u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u044f xls \u0438\u043b\u0438 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u043b\u044c\u043d\u044b\u0439 \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442 (html \u0441 \u0440\u0430\u0441\u0448\u0438\u0440\u0435\u043d\u0438\u0435\u043c .doc \u0438\u043b\u0438 .xls)<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0422\u0430\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u0432\u0438\u0434\u0436\u0435\u0442<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class WidgetTableManager(object):     u''' \u0423\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u043e\u0439 \u0441 \u0442\u0430\u0431\u043b\u0438\u0447\u043d\u044b\u043c \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u043c '''      def __init__(self, request, widget_obj, dataset, env):         u''' \u041a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 '''          self._request = request         self._widget_obj = widget_obj         self._dataset = dataset         self._env = env         self._xml = None          if widget_obj.xml_settings:             self._xml = etree.fromstring(self._env.render(widget_obj.xml_settings).replace('[[', '{{').replace(']]', '}}'))      def get_html(self):         u''' \u0412\u0435\u0440\u043d\u0443\u0442\u044c html-\u043a\u043e\u0434 \u0442\u0430\u0431\u043b\u0438\u0446\u044b '''          (fields, data) = self._dataset.get_data()          field_settings = {}         table_settings = {}         if self._xml is not None:              table_settings_node = xml_node(self._xml, '\/xml\/table')             if table_settings_node is not None:                 table_settings = table_settings_node.attrib             for xml in self._xml.xpath('\/xml\/fields\/field'):                 xml_attributes = dict(xml.attrib)                 field_name = xml_attributes['name']                  if 'lnk' in xml_attributes:                     xml_attributes['tpl_lnk'] = Template(force_unicode(xml_attributes['lnk']))                 if 'cell_attributes' in xml_attributes:                     xml_attributes['tpl_cell_attributes'] = Template(force_unicode(xml_attributes['cell_attributes']))                  field_settings[field_name] = xml_attributes          # \u0412\u044b\u0432\u043e\u0434\u0438\u043c\u044b\u0435 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d \u043a\u043e\u043b\u043e\u043d\u043a\u0438         fields_visible = []         for index, field_name in enumerate(fields):             settings = field_settings.get(field_name, {})             if 'display' in settings and settings['display'] == '0':                 continue             fields_visible.append((index, field_name, settings))          # \u0412\u044b\u0447\u0438\u0441\u043b\u0438\u0442\u044c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0438 \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u0442\u044c \u043a \u0434\u0430\u043d\u043d\u044b\u043c         rows = []         for row in data:             row_dict = dict(zip(fields, row))             row_settings = {}              # \u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0443\u0440\u043e\u0432\u043d\u044f \u0441\u0442\u0440\u043e\u043a\u0438             if 'field_row_style' in table_settings:                 row_settings['row_style'] = row_dict[table_settings['field_row_style']]             if 'field_row_attributes' in table_settings:                 row_settings['row_attributes'] = row_dict[table_settings['field_row_attributes']]              # \u041f\u0435\u0440\u0435\u0431\u0440\u0430\u0442\u044c \u0441\u0442\u0440\u043e\u043a\u0438 \u0438 \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0443\u0440\u043e\u0432\u043d\u044f \u0441\u0442\u0440\u043e\u043a\u0438             fields_set = []             for index, field_name, settings in fields_visible:                 field = {'name': field_name, 'value': row[index]}                  # \u0415\u0441\u043b\u0438 \u0435\u0441\u0442\u044c \u0441\u0441\u044b\u043b\u043a\u0430 \u0438 \u043e\u043d\u0430 \u0434\u043e\u043b\u0436\u043d\u0430 \u0431\u044b\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u0430\u043d\u0430                 if 'tpl_lnk' in settings and ('lnk_enable_field' not in settings or row_dict[settings['lnk_enable_field']] not in (0, '0', '', None)):                     field['lnk'] = self._env.render_template_compiled(settings['tpl_lnk'], row=row_dict)                  # \u0410\u0442\u0442\u0440\u0438\u0431\u0443\u0442\u044b \u044f\u0447\u0435\u0439\u043a\u0438                 if 'tpl_cell_attributes' in settings:                     field['cell_attributes'] = settings['tpl_cell_attributes'].render(Context(row_dict))                  field['settings'] = settings                 fields_set.append(field)              rows.append({'settings': row_settings, 'fields': fields_set})          return render_to_string('reports_widget_table.html', {'fields': fields_visible, 'rows': rows, 'widget_obj': self._widget_obj})      def __call__(self):         u''' \u041f\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0435, \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0442\u0430\u0431\u043b\u0438\u0446\u0443 '''          try:             return self.get_html()         except Exception as e:             return u'\u041e\u0448\u0438\u0431\u043a\u0430 \u0432 \u0432\u0438\u0434\u0436\u0435\u0442\u0435 %s: &quot;%s&quot;' % (self._widget_obj.title, e)  <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0411\u0435\u0440\u0435\u0442 \u0434\u0430\u043d\u043d\u044b\u0435 \u0438\u0437 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430 \u0434\u0430\u043d\u043d\u044b\u0445 \u0438 \u0432\u044b\u0432\u043e\u0434\u0438\u0442 \u0432 \u0442\u0430\u0431\u043b\u0438\u0447\u043d\u043e\u043c \u0432\u0438\u0434\u0435. \u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 <br \/>   \u2014 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435<br \/>   \u2014 \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e \u0441\u0441\u044b\u043b\u043e\u043a (\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043c\u043e\u0433\u0443\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0434\u043b\u044f \u0434\u0435\u0442\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u044f\u0447\u0435\u0439\u043a\u0438 \u0432 \u0434\u0440\u0443\u0433\u043e\u0439 \u0442\u0430\u0431\u043b\u0438\u0446\u0435, \u0433\u0440\u0430\u0444\u0438\u043a \u0438\u043b\u0438 excel)<br \/>   \u2014 \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u043b\u044c\u043d\u044b\u0445 \u0430\u0442\u0442\u0440\u0438\u0431\u0443\u0442\u043e\u0432 \u0441\u0442\u0440\u043e\u043a (\u0434\u043b\u044f \u0440\u0430\u0431\u043e\u0442\u044b javascript, \u0441\u043a\u0440\u044b\u0442\u0438\u044f \u0438\u043b\u0438 \u043f\u043e\u043a\u0430\u0437\u0430 \u0438\u0442\u043e\u0433\u043e\u0432\u044b\u0445 \u0441\u0442\u0440\u043e\u043a \u0438 \u0442.\u043f.)<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u041c\u0435\u043d\u0435\u0434\u0436\u0435\u0440 \u043e\u0442\u0447\u0435\u0442\u043e\u0432<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"python\">class ReportManager(object):     u''' \u0423\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442 \u043e\u0442\u0447\u0435\u0442\u0430\u043c\u0438 '''      def __init__(self, request, report):          self._report = report         self._request = request         self._user = request.user         self._forms = {}         self._env = EnvirementManager(request=request, user_params=self._get_user_params(), forms={}, f={})         self._datasets = {}         self._widgets_table = {}         self._widgets_chart = {}          self._load_stored_filter_values()          # \u0421\u0431\u043e\u0440\u043a\u0430 \u0444\u0438\u043b\u044c\u0442\u0440\u043e\u0432         for filter_obj in self._report.forms.only('title', 'html_layout'):             filter_manager = FilterManager(self._request, filter_obj, self._env)             self._save_stored_filter_values(filter_manager.get_changed_data())             self._forms[filter_obj.title] = filter_manager         self._env['forms'] = self._forms          # \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u0434\u0430\u043d\u043d\u044b\u0445         for ds in self._report.datasets.only('sql', 'title', 'xml_settings'):             self._datasets[ds.title] = DatasetManager(request, ds, self._env)         self._env['datasets'] = self._datasets          # \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0432\u0438\u0434\u0436\u0435\u0442-\u0442\u0430\u0431\u043b\u0438\u0446\u044b         for widget_obj in self._report.widgets_table.only('title', 'dataset', 'table_header', 'xml_settings'):             self._widgets_table[widget_obj.title] = WidgetTableManager(self._request,                                                                        widget_obj,                                                                        self._datasets[widget_obj.dataset.title],                                                                        self._env)         self._env['tables'] = self._widgets_table          # \u0412\u0438\u0434\u0436\u0435\u0442\u044b - \u0433\u0440\u0430\u0444\u0438\u043a\u0438         for chart_obj in self._report.widgets_chart.only('title', 'dataset', 'xml_settings'):             self._widgets_chart[chart_obj.title] = WidgetChartManager(self._request,                                                                       chart_obj,                                                                       self._datasets[chart_obj.dataset.title],                                                                       self._env)         self._env['charts'] = self._widgets_chart      def get_request(self):         u''' \u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u043e\u0442\u0447\u0435\u0442\u0430 '''          response_type = self._request.REQUEST.get('response_type', 'html')          if response_type == 'xls':             return self._get_request_xls(self._request.REQUEST['xls'])         elif response_type == 'template':             return self._get_request_template(self._request.REQUEST['template'])         else:             return self._get_request_html()      def _get_request_html(self):         u''' \u0412\u0435\u0440\u043d\u0443\u0442\u044c \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432 \u0432\u0438\u0434\u0435 html \u0434\u043b\u044f \u0432\u044b\u0432\u043e\u0434\u0430 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d '''          context = {'favorite_reports': self._user.reports_favorited.all()}         context['report_body'] = self._env.render(self._report.html_layout)         context['breadcrumbs'] = (('\u041e\u0442\u0447\u0435\u0442\u044b', reverse('reports_home')), (self._report.title, None))         context['filter_presets'] = self._report.filter_presets.filter(user=self._user)         context['report'] = self._report          return render(self._request, 'reports_report.html', context)      def _get_request_template(self):         u''' \u0412\u0435\u0440\u043d\u0443\u0442\u044c \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432 \u0432\u0438\u0434\u0435 html \u0434\u043b\u044f \u0432\u044b\u0432\u043e\u0434\u0430 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d '''          # TODO         raise NotImplementedError(u'\u041d\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u043e')      def _get_request_xls(self, dataset_title):         u&quot;&quot;&quot; \u0412\u0435\u0440\u043d\u0443\u0442\u044c \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432\u044b\u0433\u0440\u0443\u0437\u043a\u0438 \u0432 xls &quot;&quot;&quot;          dataset = self._datasets[dataset_title]         (columns, data) = dataset.get_data()         w = Workbook(optimized_write=True)         sheet = w.create_sheet(0)         sheet.append(columns)          rows_in_sheet = 0         for row in data:              if rows_in_sheet &gt; 1000000:                 sheet = w.create_sheet()                 sheet.append(columns)                 rows_in_sheet = 0              sheet.append(row)             rows_in_sheet += 1          try:             tmpFileName = os.tempnam()             w.save(tmpFileName)             fh = open(tmpFileName, 'rb')             resp = HttpResponse(fh.read(), 'application\/vnd.openxmlformats-officedocument.spreadsheetml.sheet')         finally:             if fh:                 fh.close()             os.unlink(tmpFileName)          resp['Content-Disposition'] = 'attachment; filename=&quot;\u0412\u044b\u0433\u0440\u0443\u0437\u043a\u0430.xlsx&quot;'         return resp      def _get_request_document_template(self, template_title):         u&quot;&quot;&quot; \u0412\u0435\u0440\u043d\u0443\u0442\u044c \u043e\u0442\u0432\u0435\u0442, \u0441\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u0432 \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442 \u043f\u043e \u0448\u0430\u0431\u043b\u043e\u043d\u0443 &quot;&quot;&quot;          pass      def _save_stored_filter_values(self, values):         u&quot;&quot;&quot;         \u0417\u0430\u043f\u0438\u0441\u0430\u0442\u044c \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0444\u0438\u043b\u044c\u0442\u0440\u043e\u0432 \u0432 \u0431\u0430\u0437\u0443 \u0438 \u0432 _env         \u0415\u0441\u043b\u0438 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u043d\u0435 html, \u0442\u043e \u0432 \u0431\u0430\u0437\u0443 \u043d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u043c,         \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043d\u0435 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u043f\u0435\u0440\u0435\u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b \u0438 \u043d\u0435 \u043d\u0443\u0436\u043d\u043e \u0432\u043e\u0441\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0444\u0438\u043b\u044c\u0442\u0440\u043e\u0432         &quot;&quot;&quot;          for key, value in values.items():             self._env['f'][key] = value             if values.get('response_type', 'html') == 'html':                 (store_item, is_created) = models.WidgetFormStorage.objects.get_or_create(user=self._user, report=self._report, key=key)                 store_item.value = pickle.dumps(value)                 store_item.save()      def _load_stored_filter_values(self):         u&quot;&quot;&quot; \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044c \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0444\u043e\u0440\u043c, \u0441\u0447\u0438\u0442\u0430\u0432 \u0438\u0445 \u0438\u0437 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445 &quot;&quot;&quot;          for item in self._report.form_storage.all():             self._env['f'][item.key] = pickle.loads(item.value)      def _get_user_params(self):         u&quot;&quot;&quot; \u0412\u0435\u0440\u043d\u0443\u0442\u044c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f &quot;&quot;&quot;          params = {}          try:              param_string = models.UserAccess.objects.get(report=self._report, user=self._user).params             if param_string:                 for pair in param_string.split(';'):                     (key, values_str) = pair.split('=')                     values = values_str.split(',')                     params[key] = values          except Exception, e:             pass          return params  <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0418\u043c\u0435\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043e\u0434\u0438\u043d \u043f\u0443\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u043c\u0435\u0442\u043e\u0434, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 httpResponse (html, \u0432\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438\u043b\u0438 xlsx)<\/p>\n<p>  \u0412\u043e\u0442, \u0432 \u043e\u0431\u0449\u0435\u043c-\u0442\u043e \u0438 \u0432\u0441\u0435.<br \/>  \u0418\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0430\u0434\u043c\u0438\u043d\u0438\u0441\u0442\u0440\u0430\u0442\u043e\u0440\u0430 \u043e\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u043d\u0435 \u0441\u0442\u0430\u043b.<\/p>\n<p>  css \u043e\u0442 bootstrap<\/p>\n<h2>\u041d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0438 \u0438 \u043a\u0430\u0440\u0442\u0438\u043d\u043e\u043a<\/h2>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a \u0434\u0430\u043d\u043d\u044b\u0445 ds<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"sql\">select key, value, value * 0.5 as value1 from (     select 1 as key, 2000 as value union all     select 2 as key, 4000 as value union all     select 3 as key, 6000 as value union all     select 4 as key, 3000 as value union all     select 5 as key, 2000 as value union all     select 6 as key, 1000 as value ) t {% if f.doble_rows %}cross join (select 1 union all select 2) t1 {% endif %} <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a \u0434\u0430\u043d\u043d\u044b\u0445 ds1<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"sql\">select  t.key as key, t.value as value, case when key = 1 then 'background-color: #dff0d8;' end as row_style_field, case when key = 2 then 'class=&quot;error&quot;' end as row_attribute_field {% if f.add_column %}     {% for row in datasets.ds.data %}     , {{row.key}} as \u041f\u043e\u043b\u0435{{row.key}}      {% endfor %} {% endif %} from ({{datasets.ds.sql}}) t     {% if request.GET.key %} where key = [[request.GET.key]] {% endif %}     {% if f.limit %} limit [[f.limit]] {% endif %} <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  \u0422\u0430\u0431\u043b\u0438\u0447\u043d\u044b\u0439 \u0432\u0438\u0434\u0436\u0435\u0442 t:<br \/>  \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c, \u0432\u044b\u0431\u0440\u0430\u0432 \u0438\u0437 \u0441\u043f\u0438\u0441\u043a\u0430 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a ds1<br \/>  \u0438 \u043f\u0440\u043e\u043f\u0438\u0441\u0430\u0432 \u043d\u0430\u0437\u0432\u0430\u043d\u0438\u0435. \u0412\u0441\u0435 \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u043e\u0435 \u043d\u0435 \u0442\u0440\u043e\u0433\u0430\u0435\u043c.<\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0422\u0430\u0431\u043b\u0438\u0446\u0430 t1 (\u043d\u0430 ds1)<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"xml\">&lt;xml&gt;     &lt;table field_row_style='row_style_field' field_row_attributes='row_attribute_field'\/&gt;     &lt;fields&gt;         &lt;field name='\u041a\u043b\u044e\u0447' classes='text-right'\/&gt;         &lt;field name='value' classes='text-right' lnk='?response_type=xls&amp;xls=ds&amp;key=[[row.\u041a\u043b\u044e\u0447]]&amp;from=value'\/&gt;         {% for row in datasets.ds.data %}         &lt;field name='\u041f\u043e\u043b\u0435{{row.key}}' classes='text-right' lnk='?response_type=xls&amp;xls=ds1&amp;key=[[row.\u041a\u043b\u044e\u0447]]&amp;from=\u041f\u043e\u043b\u0435{{row.key}}'\/&gt;         {% endfor %}         &lt;field name='\u041f\u043e\u043b\u04351' display='0'\/&gt;         &lt;field name='row_style_field' display='0'\/&gt;         &lt;field name='row_attribute_field' display='0'\/&gt;     &lt;\/fields&gt; &lt;\/xml&gt; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0422\u0430\u0431\u043b\u0438\u0446\u0430 t1 (\u043d\u0430 ds1)<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"xml\">&lt;xml&gt;     &lt;table field_row_style='row_style_field' field_row_attributes='row_attribute_field'\/&gt;     &lt;fields&gt;         &lt;field name='\u041a\u043b\u044e\u0447' classes='text-right'\/&gt;         &lt;field name='value' classes='text-right' lnk='?response_type=xls&amp;xls=ds&amp;key=[[row.\u041a\u043b\u044e\u0447]]&amp;from=value'\/&gt;         {% for row in datasets.ds.data %}         &lt;field name='\u041f\u043e\u043b\u0435{{row.key}}' classes='text-right' lnk='?response_type=xls&amp;xls=ds1&amp;key=[[row.\u041a\u043b\u044e\u0447]]&amp;from=\u041f\u043e\u043b\u0435{{row.key}}'\/&gt;         {% endfor %}         &lt;field name='\u041f\u043e\u043b\u04351' display='0'\/&gt;         &lt;field name='row_style_field' display='0'\/&gt;         &lt;field name='row_attribute_field' display='0'\/&gt;     &lt;\/fields&gt; &lt;\/xml&gt; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0413\u0440\u0430\u0444\u0438\u043a test (\u043d\u0430 ds)<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"xml\">&lt;xml&gt;      &lt;chart&gt;         &lt;categoryField&gt;key&lt;\/categoryField&gt;         &lt;marginTop&gt;32&lt;\/marginTop&gt;     &lt;\/chart&gt;      &lt;categoryAxis&gt;         &lt;labelsEnabled&gt;true&lt;\/labelsEnabled&gt;         &lt;gridCount&gt;50&lt;\/gridCount&gt;         &lt;equalSpacing&gt;true&lt;\/equalSpacing&gt;     &lt;\/categoryAxis&gt;      &lt;valueAxis&gt;         &lt;valueAxisLeft&gt;             &lt;stackType&gt;regular&lt;\/stackType&gt;              &lt;gridAlpha&gt;0.07&lt;\/gridAlpha&gt;         &lt;\/valueAxisLeft&gt;     &lt;\/valueAxis&gt;      &lt;cursor&gt;         &lt;bulletsEnabled&gt;true&lt;\/bulletsEnabled&gt;     &lt;\/cursor&gt;      &lt;graphs&gt;         &lt;graph&gt;             &lt;type&gt;column&lt;\/type&gt;             &lt;title&gt;\u041e\u0442\u043a\u0430\u0437\u044b&lt;\/title&gt;             &lt;valueField&gt;value&lt;\/valueField&gt;             &lt;balloonText&gt;[[category]] \u0434\u043d\u0435\u0439: [[value]] \u0448\u0442.&lt;\/balloonText&gt;             &lt;lineAlpha&gt;0&lt;\/lineAlpha&gt;             &lt;fillAlphas&gt;0.6&lt;\/fillAlphas&gt;         &lt;\/graph&gt;         &lt;graph1&gt;             &lt;type&gt;column&lt;\/type&gt;             &lt;title&gt;\u041d\u0435 \u043e\u0442\u043a\u0430\u0437\u044b&lt;\/title&gt;             &lt;valueField&gt;value1&lt;\/valueField&gt;             &lt;balloonText&gt;[[category]] \u0434\u043d\u0435\u0439: [[value]] \u0448\u0442.&lt;\/balloonText&gt;             &lt;lineAlpha&gt;0&lt;\/lineAlpha&gt;             &lt;fillAlphas&gt;0.6&lt;\/fillAlphas&gt;         &lt;\/graph1&gt;     &lt;\/graphs&gt;  &lt;\/xml&gt; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0413\u0440\u0430\u0444\u0438\u043a test1 (\u043d\u0430 ds)<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"xml\">&lt;xml&gt;      &lt;chart&gt;         &lt;categoryField&gt;key&lt;\/categoryField&gt;         &lt;marginTop&gt;32&lt;\/marginTop&gt;     &lt;\/chart&gt;      &lt;categoryAxis&gt;         &lt;labelsEnabled&gt;true&lt;\/labelsEnabled&gt;         &lt;gridCount&gt;50&lt;\/gridCount&gt;         &lt;equalSpacing&gt;true&lt;\/equalSpacing&gt;     &lt;\/categoryAxis&gt;      &lt;valueAxis&gt;         &lt;valueAxisLeft&gt;             &lt;gridAlpha&gt;0.07&lt;\/gridAlpha&gt;         &lt;\/valueAxisLeft&gt;     &lt;\/valueAxis&gt;      &lt;cursor&gt;         &lt;bulletsEnabled&gt;true&lt;\/bulletsEnabled&gt;     &lt;\/cursor&gt;      &lt;graphs&gt;         &lt;graph&gt;             &lt;type&gt;column&lt;\/type&gt;             &lt;title&gt;\u041e\u0442\u043a\u0430\u0437\u044b&lt;\/title&gt;             &lt;valueField&gt;value&lt;\/valueField&gt;             &lt;balloonText&gt;[[category]] \u0434\u043d\u0435\u0439: [[value]] \u0448\u0442.&lt;\/balloonText&gt;             &lt;lineAlpha&gt;0&lt;\/lineAlpha&gt;             &lt;fillAlphas&gt;0.6&lt;\/fillAlphas&gt;         &lt;\/graph&gt;         &lt;graph1&gt;             &lt;type&gt;column&lt;\/type&gt;             &lt;title&gt;\u041d\u0435 \u043e\u0442\u043a\u0430\u0437\u044b&lt;\/title&gt;             &lt;valueField&gt;value1&lt;\/valueField&gt;             &lt;balloonText&gt;[[category]] \u0434\u043d\u0435\u0439: [[value]] \u0448\u0442.&lt;\/balloonText&gt;             &lt;lineAlpha&gt;0&lt;\/lineAlpha&gt;             &lt;fillAlphas&gt;0.6&lt;\/fillAlphas&gt;         &lt;\/graph1&gt;         &lt;graph2&gt;             &lt;type&gt;smoothedLine&lt;\/type&gt;             &lt;title&gt;\u041d\u0435 \u043e\u0442\u043a\u0430\u0437\u044b&lt;\/title&gt;             &lt;valueField&gt;value1&lt;\/valueField&gt;         &lt;\/graph2&gt;     &lt;\/graphs&gt;  &lt;\/xml&gt; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0420\u0430\u0437\u043c\u0435\u0449\u0430\u0435\u043c \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"xml\">&lt;h2&gt;\u0420\u0430\u0431\u043e\u0442\u0430 \u0441 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0430\u043c\u0438 \u0434\u0430\u043d\u043d\u044b\u0445&lt;\/h2&gt; &lt;div class='well well-small'&gt;{{forms.f}}&lt;\/div&gt;  &lt;h3&gt;\u041f\u043e\u0441\u043b\u0435 \u043f\u0440\u0435-\u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u0440\u0430 django template&lt;\/h3&gt; &lt;pre&gt; sql:{{datasets.ds1.render.sql}} \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b:{{datasets.ds1.render.binds}} &lt;\/pre&gt;  &lt;h3&gt;\u041f\u043e\u0441\u043b\u0435 \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f&lt;\/h3&gt; {{tables.t}} &lt;a class='btn btn-success' href='?response_type=xls&xls=ds1'&gt;\u0412 Excel&lt;\/a&gt;  &lt;h2&gt;\u0412\u0438\u0437\u0443\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f \u0434\u0430\u043d\u043d\u044b\u0445&lt;\/h2&gt; {{tables.t1}} &lt;div style='height:500px;width:500px;'&gt;{{charts.test}}&lt;\/div&gt; &lt;div style='height:500px;width:500px;'&gt;{{charts.test1}}&lt;\/div&gt; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0410\u0434\u043c\u0438\u043d\u043a\u0430 \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0430\u043a<\/b><\/p>\n<div class=\"spoiler_text\"><img decoding=\"async\" src=\"https:\/\/bitbucket.org\/dibrovsd\/py_docflow\/downloads\/admin_form.png\"\/>  <\/div>\n<\/div>\n<div class=\"spoiler\"><b class=\"spoiler_title\">\u0412\u043e\u0442 \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c<\/b><\/p>\n<div class=\"spoiler_text\">\u0421\u043f\u0438\u0441\u043e\u043a \u043e\u0442\u0447\u0435\u0442\u043e\u0432:<br \/>  <img decoding=\"async\" src=\"https:\/\/bitbucket.org\/dibrovsd\/py_docflow\/downloads\/report_list.png\"\/><\/p>\n<p>  \u041d\u0430\u0448 \u043e\u0442\u0447\u0435\u0442:<br \/>  <img decoding=\"async\" src=\"https:\/\/bitbucket.org\/dibrovsd\/py_docflow\/downloads\/all_page.png\"\/>  <\/div>\n<\/div>\n<p>  \u0425\u043e\u0447\u0443 \u0441\u043a\u0430\u0437\u0430\u0442\u044c \u0441\u043f\u0430\u0441\u0438\u0431\u043e \u0437\u0430 \u043f\u043e\u043c\u043e\u0449\u044c \u041f\u0430\u0432\u043b\u0443, \u0410\u043b\u0435\u043a\u0441\u0430\u043d\u0434\u0440\u0443, \u0415\u0432\u0433\u0435\u043d\u0438\u044e.<br \/>  \u0421\u043f\u0430\u0441\u0438\u0431\u043e \u0437\u0430 \u0432\u043d\u0438\u043c\u0430\u043d\u0438\u0435. \u0415\u0441\u043b\u0438 \u0445\u043e\u0442\u0438\u0442\u0435, \u043c\u043e\u0436\u0435\u0442\u0435 \u0432\u0437\u044f\u0442\u044c \u044d\u0442\u043e \u0438\u0437 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u044f \u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u043f\u043e \u0441\u0432\u043e\u0435\u043c\u0443 \u0443\u0441\u043c\u043e\u0442\u0440\u0435\u043d\u0438\u044e.<br \/>  <a href=\"https:\/\/bitbucket.org\/dibrovsd\/py_docflow\/\">bitbucket.org\/dibrovsd\/py_docflow\/<\/a><\/p>\n<p>  P.S. \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438 \u0435\u0441\u0442\u044c \u0435\u0449\u0435 \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u0437\u0430\u043a\u043e\u043d\u0447\u0435\u043d\u043d\u043e\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u043e\u043e\u0431\u043e\u0440\u043e\u0442\u0430, <br \/>  \u043d\u043e \u043e \u043d\u0435\u043c \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u043e\u0437\u0436\u0435, \u0435\u0441\u043b\u0438 \u0432\u0430\u043c \u044d\u0442\u043e \u0431\u0443\u0434\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e.<\/p>\n<p>  P.P.S.<br \/>  \u041d\u0430\u0432\u0435\u0440\u043d\u044f\u043a\u0430 \u043d\u0430\u0439\u0434\u0435\u0442\u0441\u044f \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u043e \u043d\u0435 \u043e\u043f\u0442\u0438\u043c\u0430\u043b\u044c\u043d\u044b\u0445 \u0440\u0435\u0448\u0435\u043d\u0438\u0439, \u0441 python \u0438 django \u044f \u0437\u043d\u0430\u043a\u043e\u043c \u043d\u0435\u0434\u0430\u0432\u043d\u043e.<br \/>  \u0412\u0441\u0435 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u0438\u0432\u043d\u044b\u0435 \u043f\u0440\u0435\u0434\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0448\u0443 \u043f\u0438\u0441\u0430\u0442\u044c \u0432 \u00ab\u043b\u0438\u0447\u043a\u0443\u00bb.    \t \t\t   \t<\/p>\n<div class=\"clear\"><\/div>\n<\/p><\/div>\n<p> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"http:\/\/habrahabr.ru\/post\/177567\/\"> http:\/\/habrahabr.ru\/post\/177567\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div class=\"content html_format\">   \t\u0414\u043e\u0431\u0440\u043e\u0433\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0441\u0443\u0442\u043e\u043a.<\/p>\n<p>  \u0422\u0430\u043a \u0441\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c, \u0447\u0442\u043e \u043c\u043e\u044f \u0440\u0430\u0431\u043e\u0442\u0430 \u0441\u0432\u044f\u0437\u0430\u043d\u0430 \u0441 \u043d\u0430\u043f\u0438\u0441\u0430\u043d\u0438\u0435\u043c \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<br \/>  \u042d\u0442\u043e\u043c\u0443 \u044f \u043f\u043e\u0441\u0432\u044f\u0442\u0438\u043b \u043e\u043a\u043e\u043b\u043e 8 \u043b\u0435\u0442. \u041e\u0442\u0447\u0435\u0442\u044b \u2014 \u044d\u0442\u043e \u0433\u043b\u0430\u0437\u0430 \u0431\u0438\u0437\u043d\u0435\u0441-\u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0430 \u0438 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f, <br \/>  \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u0430\u044f \u0434\u043b\u044f \u043f\u0440\u0438\u043d\u044f\u0442\u0438\u044f \u043e\u043f\u0435\u0440\u0430\u0442\u0438\u0432\u043d\u044b\u0445 \u0440\u0435\u0448\u0435\u043d\u0438\u0439.<\/p>\n<p>  \u0412\u043d\u0430\u0447\u0430\u043b\u0435 \u043d\u0430\u0448 \u043e\u0442\u0434\u0435\u043b \u0434\u0435\u043b\u0430\u043b \u043e\u0442\u0447\u0435\u0442\u044b, <br \/>   \u2014 \u041f\u0440\u0438\u043d\u0438\u043c\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0438 \u043f\u043e outlook<br \/>   \u2014 \u0421\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u044f sql-\u0437\u0430\u043f\u0440\u043e\u0441<br \/>   \u2014 \u041e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u044f \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b \u0437\u0430\u043a\u0430\u0437\u0447\u0438\u043a\u0443 \u0432 xls<br \/>   \u2014 \u0412 \u043b\u0443\u0447\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435, \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u044f sql-\u043a\u043e\u0434 \u043a\u0443\u0434\u0430-\u0442\u043e \u0432 \u043f\u0430\u043f\u043a\u0443 (\u0430 \u0438\u043d\u043e\u0433\u0434\u0430 \u0438 \u043d\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u044f)<\/p>\n<p>  \u041d\u043e \u044d\u0442\u043e \u0431\u044b\u043b\u043e \u0441\u043a\u0443\u0447\u043d\u043e \u0438 \u043d\u0435\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e. \u0422\u0430\u043a \u043f\u043e\u044f\u0432\u0438\u043b\u043e\u0441\u044c \u043f\u0440\u043e\u0441\u0442\u0435\u0439\u0448\u0435\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043d\u0430 PHP, <br \/>  \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043a\u0430\u0436\u0434\u044b\u0439 \u043e\u0442\u0447\u0435\u0442 \u0431\u044b\u043b \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u0435\u043d \u0432 \u0432\u0438\u0434\u0435 php-\u0444\u0430\u0439\u043b\u0430 \u0441 \u043e\u0434\u043d\u0438\u043c \u043a\u043b\u0430\u0441\u0441\u043e\u043c, \u0438\u043c\u0435\u044e\u0449\u0438\u043c \u0435\u0434\u0438\u043d\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 (\u043f\u043e\u043c\u0438\u043c\u043e \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0430) \u043c\u0435\u0442\u043e\u0434 show()<\/p>\n<p>  \u0412 \u0442\u0430\u043a\u043e\u043c \u0432\u0438\u0434\u0435, \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u043f\u0440\u043e\u0436\u0438\u043b\u0430 5,5 \u043b\u0435\u0442, \u0437\u0430 \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043c\u043d\u043e\u0439 \u0438 \u0435\u0449\u0435 \u043e\u0434\u043d\u0438\u043c \u0447\u0435\u043b\u043e\u0432\u0435\u043a\u043e\u043c \u0431\u044b\u043b\u043e \u043d\u0430\u043f\u0438\u0441\u0430\u043d\u043e \u0431\u043e\u043b\u0435\u0435 500 \u0440\u0430\u0437\u043b\u0438\u0447\u043d\u044b\u0445 \u043e\u0442\u0447\u0435\u0442\u043e\u0432.<br \/>  \u0412 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u043f\u043e\u044f\u0432\u0438\u043b\u0441\u044f \u043e\u043f\u044b\u0442 \u0438 \u0441\u0442\u0430\u043b\u043e \u043f\u043e\u043d\u044f\u0442\u043d\u043e, \u0447\u0442\u043e \u043c\u043d\u043e\u0433\u043e\u0435 (\u0435\u0441\u043b\u0438 \u043d\u0435 \u0432\u0441\u0435) \u0441\u0434\u0435\u043b\u0430\u043d\u043e \u043d\u0435 \u0442\u0430\u043a, \u0434\u0430 \u0438 PHP \u0443\u0436\u0435 \u043d\u0435 \u0443\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u043b.<\/p>\n<p>  \u0421\u0435\u0440\u0432\u0435\u0440 \u043e\u0442\u0447\u0435\u0442\u043e\u0432 \u0431\u044b\u043b \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u043d \u043d\u0430 django, \u0433\u0434\u0435 \u0431\u044b\u043b\u0430 \u00ab\u0430\u0434\u043c\u0438\u043d\u043a\u0430\u00bb \u0438 \u043a\u043e\u0434 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0443\u0436\u0435 \u043d\u0435 \u043f\u0440\u0430\u0432\u0438\u043b\u0441\u044f.<br \/>  \u0412 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0440\u0430\u0431\u043e\u0442\u044b \u0441\u043d\u043e\u0432\u0430 \u043d\u0430\u043a\u043e\u043f\u0438\u043b\u043e\u0441\u044c \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043c\u044b\u0441\u043b\u0435\u0439, <br \/>  \u0432 \u0438\u0442\u043e\u0433\u0435 \u0441\u0435\u0440\u0432\u0435\u0440 \u0431\u044b\u043b \u0441\u043d\u043e\u0432\u0430 \u043f\u0435\u0440\u0435\u043f\u0438\u0441\u0430\u043d.<\/p>\n<p>  \u042f \u0440\u0435\u0448\u0438\u043b \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0441\u043e\u043e\u0431\u0449\u0435\u0441\u0442\u0432\u0443 \u0442\u0435\u043a\u0443\u0449\u0443\u044e \u0432\u0435\u0440\u0441\u0438\u044e \u0441\u0438\u0441\u0442\u0435\u043c\u044b, \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e, \u043e\u043d\u0430 \u043a\u043e\u043c\u0443-\u0442\u043e \u043e\u0431\u043b\u0435\u0433\u0447\u0438\u0442 \u0436\u0438\u0437\u043d\u044c,<br \/>  \u0441\u043d\u044f\u0432 \u0440\u0443\u0442\u0438\u043d\u0443 \u043e\u0442\u0447\u0435\u0442\u043d\u043e\u0441\u0442\u0438 \u0438 \u043f\u0435\u0440\u0435\u043b\u043e\u0436\u0438\u0442 \u0435\u0435 \u043d\u0430 \u043c\u0430\u0448\u0438\u043d\u0443, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0433\u043e\u0442\u043e\u0432\u0430 \u0441\u043e\u0442\u043d\u0438 \u0440\u0430\u0437 \u0444\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043e\u0442\u0447\u0435\u0442\u044b<br \/>  \u0442\u044b\u0441\u044f\u0447\u0438 \u043b\u044e\u0434\u044f\u043c \u043f\u043e \u0441\u043e\u0442\u043d\u0438 \u0440\u0430\u0437\u043d\u044b\u0445 \u0432\u0445\u043e\u0434\u043d\u044b\u0445 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u0432.<\/p>\n<p>  \u042d\u0442\u043e \u0440\u0430\u0431\u043e\u0447\u0438\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u044b \u043c\u043e\u0436\u0435\u0442\u0435 (\u0435\u0441\u043b\u0438 \u0437\u0430\u0445\u043e\u0442\u0438\u0442\u0435) \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432 \u0440\u0430\u0431\u043e\u0442\u0435,<br \/>  \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043d\u0430\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u043f\u0440\u043e\u0441\u0442\u043e \u043d\u0440\u0430\u0432\u0438\u0442\u0441\u044f.<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-177567","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/177567","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=177567"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/177567\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=177567"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=177567"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=177567"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}