Completed
Push — fix_travis ( e39dc3 )
by Ryo
315:01 queued 308:56
created

ProductController   F

Complexity

Total Complexity 80

Size/Duplication

Total Lines 781
Duplicated Lines 11.01 %

Coupling/Cohesion

Components 0
Dependencies 12

Test Coverage

Coverage 64.43%

Importance

Changes 0
Metric Value
dl 86
loc 781
ccs 279
cts 433
cp 0.6443
rs 1.0434
c 0
b 0
f 0
wmc 80
lcom 0
cbo 12

8 Methods

Rating   Name   Duplication   Size   Complexity  
B addImage() 0 38 6
F edit() 27 284 34
C delete() 0 89 11
C copy() 0 94 9
A display() 0 14 2
D index() 7 141 13
B export() 41 87 4
A createProductCategory() 11 11 1

How to fix   Duplicated Code    Complexity   

Duplicated Code

Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.

Common duplication problems, and corresponding solutions are:

Complex Class

 Tip:   Before tackling complexity, make sure that you eliminate any duplication first. This often can reduce the size of classes significantly.

Complex classes like ProductController often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.

Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.

While breaking up the class, it is a good idea to analyze how other classes use ProductController, and based on these observations, apply Extract Interface, too.

1
<?php
2
/*
3
 * This file is part of EC-CUBE
4
 *
5
 * Copyright(c) 2000-2015 LOCKON CO.,LTD. All Rights Reserved.
6
 *
7
 * http://www.lockon.co.jp/
8
 *
9
 * This program is free software; you can redistribute it and/or
10
 * modify it under the terms of the GNU General Public License
11
 * as published by the Free Software Foundation; either version 2
12
 * of the License, or (at your option) any later version.
13
 *
14
 * This program is distributed in the hope that it will be useful,
15
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17
 * GNU General Public License for more details.
18
 *
19
 * You should have received a copy of the GNU General Public License
20
 * along with this program; if not, write to the Free Software
21
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
 */
23
24
25
namespace Eccube\Controller\Admin\Product;
26
27
use Eccube\Application;
28
use Eccube\Common\Constant;
29
use Eccube\Controller\AbstractController;
30
use Eccube\Entity\Master\CsvType;
31
use Eccube\Entity\ProductTag;
32
use Eccube\Event\EccubeEvents;
33
use Eccube\Event\EventArgs;
34
use Symfony\Component\Filesystem\Filesystem;
35
use Symfony\Component\HttpFoundation\File\File;
36
use Symfony\Component\HttpFoundation\Request;
37
use Symfony\Component\HttpFoundation\StreamedResponse;
38
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
39
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
40
use Symfony\Component\HttpKernel\Exception\UnsupportedMediaTypeHttpException;
41
42
class ProductController extends AbstractController
0 ignored issues
show
introduced by
Missing class doc comment
Loading history...
43
{
44 7
    public function index(Application $app, Request $request, $page_no = null)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
45
    {
46
47 7
        $session = $app['session'];
48
49 7
        $builder = $app['form.factory']
50 7
            ->createBuilder('admin_search_product');
51
52 7
        $event = new EventArgs(
53
            array(
54 7
                'builder' => $builder,
55
            ),
56
            $request
57
        );
58 7
        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_INDEX_INITIALIZE, $event);
59
60 7
        $searchForm = $builder->getForm();
61
62 7
        $pagination = array();
63
64 7
        $disps = $app['eccube.repository.master.disp']->findAll();
65 7
        $pageMaxis = $app['eccube.repository.master.page_max']->findAll();
66 7
        $page_count = $app['config']['default_page_count'];
67 7
        $page_status = null;
68 7
        $active = false;
69
70 7
        if ('POST' === $request->getMethod()) {
0 ignored issues
show
Coding Style introduced by
Blank line found at start of control structure
Loading history...
71
72 5
            $searchForm->handleRequest($request);
73
74 5
            if ($searchForm->isValid()) {
75 5
                $searchData = $searchForm->getData();
76
77
                // paginator
78 5
                $qb = $app['eccube.repository.product']->getQueryBuilderBySearchDataForAdmin($searchData);
79 5
                $page_no = 1;
80
81 5
                $event = new EventArgs(
82
                    array(
83 5
                        'qb' => $qb,
84 5
                        'searchData' => $searchData,
85
                    ),
86
                    $request
87
                );
88 5
                $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_INDEX_SEARCH, $event);
89 5
                $searchData = $event->getArgument('searchData');
90
91 5
                $pagination = $app['paginator']()->paginate(
92
                    $qb,
93
                    $page_no,
94
                    $page_count,
95 5
                    array('wrap-queries' => true)
96
                );
97
98
                // sessionのデータ保持
99 5
                $session->set('eccube.admin.product.search', $searchData);
100 5
                $session->set('eccube.admin.product.search.page_no', $page_no);
101
            }
102
        } else {
103 2
            if (is_null($page_no) && $request->get('resume') != Constant::ENABLED) {
104
                // sessionを削除
105 2
                $session->remove('eccube.admin.product.search');
106 2
                $session->remove('eccube.admin.product.search.page_no');
107
            } else {
108
                // pagingなどの処理
109
                $searchData = $session->get('eccube.admin.product.search');
110
                if (is_null($page_no)) {
111
                    $page_no = intval($session->get('eccube.admin.product.search.page_no'));
112
                } else {
113
                    $session->set('eccube.admin.product.search.page_no', $page_no);
114
                }
115
                if (!is_null($searchData)) {
0 ignored issues
show
Coding Style introduced by
Blank line found at start of control structure
Loading history...
116
117
                    // 公開ステータス
118
                    $status = $request->get('status');
119
                    if (!empty($status)) {
120
                        if ($status != $app['config']['admin_product_stock_status']) {
121
                            $searchData['link_status'] = $app['eccube.repository.master.disp']->find($status);
122
                            $searchData['status'] = null;
123
                            $session->set('eccube.admin.product.search', $searchData);
124
                        } else {
125
                            $searchData['stock_status'] = Constant::DISABLED;
126
                        }
127
                        $page_status = $status;
128
                    } else {
129
                        $searchData['link_status'] = null;
130
                        $searchData['stock_status'] = null;
131
                    }
132
                    // 表示件数
133
                    $pcount = $request->get('page_count');
134
135
                    $page_count = empty($pcount) ? $page_count : $pcount;
136
137
                    $qb = $app['eccube.repository.product']->getQueryBuilderBySearchDataForAdmin($searchData);
138
139
                    $event = new EventArgs(
140
                        array(
141
                            'qb' => $qb,
142
                            'searchData' => $searchData,
143
                        ),
144
                        $request
145
                    );
146
                    $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_INDEX_SEARCH, $event);
147
                    $searchData = $event->getArgument('searchData');
148
149
                    $pagination = $app['paginator']()->paginate(
150
                        $qb,
151
                        $page_no,
152
                        $page_count,
153
                        array('wrap-queries' => true)
154
                    );
155
156
                    // セッションから検索条件を復元(カテゴリ)
157
                    if (!empty($searchData['category_id'])) {
158
                        $searchData['category_id'] = $app['eccube.repository.category']->find($searchData['category_id']);
159
                    }
160
                    // セッションから検索条件を復元(スーテタス)
161 View Code Duplication
                    if (count($searchData['status']) > 0) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
162
                        $status_ids = array();
163
                        foreach ($searchData['status'] as $Status) {
164
                            $status_ids[] = $Status->getId();
165
                        }
166
                        $searchData['status'] = $app['eccube.repository.master.disp']->findBy(array('id' => $status_ids));
167
                    }
168
                    
0 ignored issues
show
introduced by
Please trim any trailing whitespace
Loading history...
169
                    $searchForm->setData($searchData);
170
                }
171
            }
172
        }
173
174 7
        return $app->render('Product/index.twig', array(
175 7
            'searchForm' => $searchForm->createView(),
176 7
            'pagination' => $pagination,
177 7
            'disps' => $disps,
178 7
            'pageMaxis' => $pageMaxis,
179 7
            'page_no' => $page_no,
180 7
            'page_status' => $page_status,
181 7
            'page_count' => $page_count,
182 7
            'active' => $active,
183
        ));
184
    }
185
186
    public function addImage(Application $app, Request $request)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
187
    {
188
        if (!$request->isXmlHttpRequest()) {
189
            throw new BadRequestHttpException('リクエストが不正です');
190
        }
191
192
        $images = $request->files->get('admin_product');
193
194
        $files = array();
195
        if (count($images) > 0) {
196
            foreach ($images as $img) {
197
                foreach ($img as $image) {
198
                    //ファイルフォーマット検証
199
                    $mimeType = $image->getMimeType();
200
                    if (0 !== strpos($mimeType, 'image')) {
201
                        throw new UnsupportedMediaTypeHttpException('ファイル形式が不正です');
202
                    }
203
204
                    $extension = $image->getClientOriginalExtension();
205
                    $filename = date('mdHis') . uniqid('_') . '.' . $extension;
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
206
                    $image->move($app['config']['image_temp_realdir'], $filename);
207
                    $files[] = $filename;
208
                }
209
            }
210
        }
211
212
        $event = new EventArgs(
213
            array(
214
                'images' => $images,
215
                'files' => $files,
216
            ),
217
            $request
218
        );
219
        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_ADD_IMAGE_COMPLETE, $event);
220
        $files = $event->getArgument('files');
221
222
        return $app->json(array('files' => $files), 200);
223
    }
224
225 19
    public function edit(Application $app, Request $request, $id = null)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
226
    {
227 19
        $has_class = false;
228 19
        if (is_null($id)) {
229 6
            $Product = new \Eccube\Entity\Product();
230 6
            $ProductClass = new \Eccube\Entity\ProductClass();
231 6
            $Disp = $app['eccube.repository.master.disp']->find(\Eccube\Entity\Master\Disp::DISPLAY_HIDE);
232
            $Product
233 6
                ->setDelFlg(Constant::DISABLED)
234 6
                ->addProductClass($ProductClass)
235 6
                ->setStatus($Disp);
236
            $ProductClass
237 6
                ->setDelFlg(Constant::DISABLED)
238 6
                ->setStockUnlimited(true)
239 6
                ->setProduct($Product);
240 6
            $ProductStock = new \Eccube\Entity\ProductStock();
241 6
            $ProductClass->setProductStock($ProductStock);
242 6
            $ProductStock->setProductClass($ProductClass);
243
        } else {
244 13
            $Product = $app['eccube.repository.product']->find($id);
245 13
            if (!$Product) {
246
                throw new NotFoundHttpException();
247
            }
248
            // 規格あり商品か
249 13
            $has_class = $Product->hasProductClass();
250 13
            if (!$has_class) {
251 11
                $ProductClasses = $Product->getProductClasses();
252 11
                $ProductClass = $ProductClasses[0];
253 11
                $BaseInfo = $app['eccube.repository.base_info']->get();
254 11 View Code Duplication
                if ($BaseInfo->getOptionProductTaxRule() == Constant::ENABLED && $ProductClass->getTaxRule() && !$ProductClass->getTaxRule()->getDelFlg()) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
255 6
                    $ProductClass->setTaxRate($ProductClass->getTaxRule()->getTaxRate());
256
                }
257 11
                $ProductStock = $ProductClasses[0]->getProductStock();
258
            }
259
        }
260
261 19
        $builder = $app['form.factory']
262 19
            ->createBuilder('admin_product', $Product);
263
264
        // 規格あり商品の場合、規格関連情報をFormから除外
265 19
        if ($has_class) {
266 2
            $builder->remove('class');
267
        }
268
269 19
        $event = new EventArgs(
270
            array(
271 19
                'builder' => $builder,
272 19
                'Product' => $Product,
273
            ),
274
            $request
275
        );
276 19
        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_EDIT_INITIALIZE, $event);
277
278 19
        $form = $builder->getForm();
279
280 19
        if (!$has_class) {
281 17
            $ProductClass->setStockUnlimited((boolean)$ProductClass->getStockUnlimited());
0 ignored issues
show
Bug introduced by
The variable $ProductClass does not seem to be defined for all execution paths leading up to this point.

If you define a variable conditionally, it can happen that it is not defined for all execution paths.

Let’s take a look at an example:

function myFunction($a) {
    switch ($a) {
        case 'foo':
            $x = 1;
            break;

        case 'bar':
            $x = 2;
            break;
    }

    // $x is potentially undefined here.
    echo $x;
}

In the above example, the variable $x is defined if you pass “foo” or “bar” as argument for $a. However, since the switch statement has no default case statement, if you pass any other value, the variable $x would be undefined.

Available Fixes

  1. Check for existence of the variable explicitly:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        if (isset($x)) { // Make sure it's always set.
            echo $x;
        }
    }
    
  2. Define a default value for the variable:

    function myFunction($a) {
        $x = ''; // Set a default which gets overridden for certain paths.
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        echo $x;
    }
    
  3. Add a value for the missing path:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
    
            // We add support for the missing case.
            default:
                $x = '';
                break;
        }
    
        echo $x;
    }
    
Loading history...
Coding Style introduced by
As per coding-style, a cast statement should be followed by a single space.
Loading history...
282 17
            $form['class']->setData($ProductClass);
283
        }
284
285
        // ファイルの登録
286 19
        $images = array();
287 19
        $ProductImages = $Product->getProductImage();
288 19
        foreach ($ProductImages as $ProductImage) {
289 19
            $images[] = $ProductImage->getFileName();
290
        }
291 19
        $form['images']->setData($images);
292
293 19
        $categories = array();
294 19
        $ProductCategories = $Product->getProductCategories();
295 19
        foreach ($ProductCategories as $ProductCategory) {
296
            /* @var $ProductCategory \Eccube\Entity\ProductCategory */
297 19
            $categories[] = $ProductCategory->getCategory();
298
        }
299 19
        $form['Category']->setData($categories);
300
301 19
        $Tags = array();
302 19
        $ProductTags = $Product->getProductTag();
303 19
        foreach ($ProductTags as $ProductTag) {
304 19
            $Tags[] = $ProductTag->getTag();
305
        }
306 19
        $form['Tag']->setData($Tags);
307
308 19
        if ('POST' === $request->getMethod()) {
309 14
            $form->handleRequest($request);
310 14
            if ($form->isValid()) {
311 14
                log_info('商品登録開始', array($id));
312 14
                $Product = $form->getData();
313
314 14
                if (!$has_class) {
315 14
                    $ProductClass = $form['class']->getData();
316
317
                    // 個別消費税
318 14
                    $BaseInfo = $app['eccube.repository.base_info']->get();
319 14
                    if ($BaseInfo->getOptionProductTaxRule() == Constant::ENABLED) {
320 12
                        if ($ProductClass->getTaxRate() !== null) {
321 8 View Code Duplication
                            if ($ProductClass->getTaxRule()) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
322 4
                                if ($ProductClass->getTaxRule()->getDelFlg() == Constant::ENABLED) {
323
                                    $ProductClass->getTaxRule()->setDelFlg(Constant::DISABLED);
324
                                }
325
326 4
                                $ProductClass->getTaxRule()->setTaxRate($ProductClass->getTaxRate());
327
                            } else {
328 4
                                $taxrule = $app['eccube.repository.tax_rule']->newTaxRule();
329 4
                                $taxrule->setTaxRate($ProductClass->getTaxRate());
330 4
                                $taxrule->setApplyDate(new \DateTime());
331 4
                                $taxrule->setProduct($Product);
332 4
                                $taxrule->setProductClass($ProductClass);
333 8
                                $ProductClass->setTaxRule($taxrule);
334
                            }
335
                        } else {
336 4
                            if ($ProductClass->getTaxRule()) {
337 2
                                $ProductClass->getTaxRule()->setDelFlg(Constant::ENABLED);
338
                            }
339
                        }
340
                    }
341 14
                    $app['orm.em']->persist($ProductClass);
342
343
                    // 在庫情報を作成
344 14
                    if (!$ProductClass->getStockUnlimited()) {
345
                        $ProductStock->setStock($ProductClass->getStock());
0 ignored issues
show
Bug introduced by
The variable $ProductStock does not seem to be defined for all execution paths leading up to this point.

If you define a variable conditionally, it can happen that it is not defined for all execution paths.

Let’s take a look at an example:

function myFunction($a) {
    switch ($a) {
        case 'foo':
            $x = 1;
            break;

        case 'bar':
            $x = 2;
            break;
    }

    // $x is potentially undefined here.
    echo $x;
}

In the above example, the variable $x is defined if you pass “foo” or “bar” as argument for $a. However, since the switch statement has no default case statement, if you pass any other value, the variable $x would be undefined.

Available Fixes

  1. Check for existence of the variable explicitly:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        if (isset($x)) { // Make sure it's always set.
            echo $x;
        }
    }
    
  2. Define a default value for the variable:

    function myFunction($a) {
        $x = ''; // Set a default which gets overridden for certain paths.
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
        }
    
        echo $x;
    }
    
  3. Add a value for the missing path:

    function myFunction($a) {
        switch ($a) {
            case 'foo':
                $x = 1;
                break;
    
            case 'bar':
                $x = 2;
                break;
    
            // We add support for the missing case.
            default:
                $x = '';
                break;
        }
    
        echo $x;
    }
    
Loading history...
346
                    } else {
347
                        // 在庫無制限時はnullを設定
348 14
                        $ProductStock->setStock(null);
349
                    }
350 14
                    $app['orm.em']->persist($ProductStock);
351
                }
352
353
                // カテゴリの登録
354
                // 一度クリア
355
                /* @var $Product \Eccube\Entity\Product */
356 14
                foreach ($Product->getProductCategories() as $ProductCategory) {
357 11
                    $Product->removeProductCategory($ProductCategory);
358 14
                    $app['orm.em']->remove($ProductCategory);
359
                }
360 14
                $app['orm.em']->persist($Product);
361 14
                $app['orm.em']->flush();
362
363 14
                $count = 1;
364 14
                $Categories = $form->get('Category')->getData();
365 14
                $categoriesIdList = array();
366 14
                foreach ($Categories as $Category) {
367 View Code Duplication
                    foreach($Category->getPath() as $ParentCategory){
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
368
                        if (!isset($categoriesIdList[$ParentCategory->getId()])){
369
                            $ProductCategory = $this->createProductCategory($Product, $ParentCategory, $count);
370
                            $app['orm.em']->persist($ProductCategory);
371
                            $count++;
372
                            /* @var $Product \Eccube\Entity\Product */
373
                            $Product->addProductCategory($ProductCategory);
374
                            $categoriesIdList[$ParentCategory->getId()] = true;
375
                        }
376
                    }
377
                    if (!isset($categoriesIdList[$Category->getId()])){
378
                        $ProductCategory = $this->createProductCategory($Product, $Category, $count);
379
                        $app['orm.em']->persist($ProductCategory);
380
                        $count++;
381
                        /* @var $Product \Eccube\Entity\Product */
382
                        $Product->addProductCategory($ProductCategory);
383 14
                        $categoriesIdList[$Category->getId()] = true;
384
                    }
385
                }
386
387
                // 画像の登録
388 14
                $add_images = $form->get('add_images')->getData();
389 14
                foreach ($add_images as $add_image) {
390
                    $ProductImage = new \Eccube\Entity\ProductImage();
391
                    $ProductImage
392
                        ->setFileName($add_image)
393
                        ->setProduct($Product)
394
                        ->setRank(1);
395
                    $Product->addProductImage($ProductImage);
396
                    $app['orm.em']->persist($ProductImage);
397
398
                    // 移動
399
                    $file = new File($app['config']['image_temp_realdir'] . '/' . $add_image);
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
400 14
                    $file->move($app['config']['image_save_realdir']);
401
                }
402
403
                // 画像の削除
404 14
                $delete_images = $form->get('delete_images')->getData();
405 14
                foreach ($delete_images as $delete_image) {
406
                    $ProductImage = $app['eccube.repository.product_image']
407
                        ->findOneBy(array('file_name' => $delete_image));
408
409
                    // 追加してすぐに削除した画像は、Entityに追加されない
410
                    if ($ProductImage instanceof \Eccube\Entity\ProductImage) {
0 ignored issues
show
Bug introduced by
The class Eccube\Entity\ProductImage does not exist. Did you forget a USE statement, or did you not list all dependencies?

This error could be the result of:

1. Missing dependencies

PHP Analyzer uses your composer.json file (if available) to determine the dependencies of your project and to determine all the available classes and functions. It expects the composer.json to be in the root folder of your repository.

Are you sure this class is defined by one of your dependencies, or did you maybe not list a dependency in either the require or require-dev section?

2. Missing use statement

PHP does not complain about undefined classes in ìnstanceof checks. For example, the following PHP code will work perfectly fine:

if ($x instanceof DoesNotExist) {
    // Do something.
}

If you have not tested against this specific condition, such errors might go unnoticed.

Loading history...
411
                        $Product->removeProductImage($ProductImage);
412
                        $app['orm.em']->remove($ProductImage);
413
0 ignored issues
show
Coding Style introduced by
Blank line found at end of control structure
Loading history...
414
                    }
415
                    $app['orm.em']->persist($Product);
416
417
                    // 削除
418
                    $fs = new Filesystem();
419 14
                    $fs->remove($app['config']['image_save_realdir'] . '/' . $delete_image);
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
420
                }
421 14
                $app['orm.em']->persist($Product);
422 14
                $app['orm.em']->flush();
423
424
425 14
                $ranks = $request->get('rank_images');
426 14
                if ($ranks) {
427
                    foreach ($ranks as $rank) {
428
                        list($filename, $rank_val) = explode('//', $rank);
429
                        $ProductImage = $app['eccube.repository.product_image']
430
                            ->findOneBy(array(
431
                                'file_name' => $filename,
432
                                'Product' => $Product,
433
                            ));
434
                        $ProductImage->setRank($rank_val);
435
                        $app['orm.em']->persist($ProductImage);
436
                    }
437
                }
438 14
                $app['orm.em']->flush();
439
440
                // 商品タグの登録
441
                // 商品タグを一度クリア
442 14
                $ProductTags = $Product->getProductTag();
443 14
                foreach ($ProductTags as $ProductTag) {
444
                    $Product->removeProductTag($ProductTag);
445 14
                    $app['orm.em']->remove($ProductTag);
446
                }
447
448
                // 商品タグの登録
449 14
                $Tags = $form->get('Tag')->getData();
450 14
                foreach ($Tags as $Tag) {
451 14
                    $ProductTag = new ProductTag();
452
                    $ProductTag
453 14
                        ->setProduct($Product)
454 14
                        ->setTag($Tag);
455 14
                    $Product->addProductTag($ProductTag);
456 14
                    $app['orm.em']->persist($ProductTag);
457
                }
458
459 14
                $Product->setUpdateDate(new \DateTime());
460 14
                $app['orm.em']->flush();
461
462 14
                log_info('商品登録完了', array($id));
463
464 14
                $event = new EventArgs(
465
                    array(
466 14
                        'form' => $form,
467 14
                        'Product' => $Product,
468
                    ),
469
                    $request
470
                );
471 14
                $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_EDIT_COMPLETE, $event);
472
473 14
                $app->addSuccess('admin.register.complete', 'admin');
474
475 14
                return $app->redirect($app->url('admin_product_product_edit', array('id' => $Product->getId())));
476
            } else {
477
                log_info('商品登録チェックエラー', array($id));
478
                $app->addError('admin.register.failed', 'admin');
479
            }
480
        }
481
482
        // 検索結果の保持
483 5
        $builder = $app['form.factory']
484 5
            ->createBuilder('admin_search_product');
485
486 5
        $event = new EventArgs(
487
            array(
488 5
                'builder' => $builder,
489 5
                'Product' => $Product,
490
            ),
491
            $request
492
        );
493 5
        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_EDIT_SEARCH, $event);
494
495 5
        $searchForm = $builder->getForm();
496
497 5
        if ('POST' === $request->getMethod()) {
498
            $searchForm->handleRequest($request);
499
        }
500
501 5
        return $app->render('Product/product.twig', array(
502 5
            'Product' => $Product,
503 5
            'form' => $form->createView(),
504 5
            'searchForm' => $searchForm->createView(),
505 5
            'has_class' => $has_class,
506 5
            'id' => $id,
507
        ));
508
    }
509
510 2
    public function delete(Application $app, Request $request, $id = null)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
511
    {
512 2
        $this->isTokenValid($app);
513 2
        $session = $request->getSession();
514 2
        $page_no = intval($session->get('eccube.admin.product.search.page_no'));
515 2
        $page_no = $page_no ? $page_no : Constant::ENABLED;
516
517 2
        if (!is_null($id)) {
518
            /* @var $Product \Eccube\Entity\Product */
519 2
            $Product = $app['eccube.repository.product']->find($id);
520 2
            if (!$Product) {
521
                $app->deleteMessage();
522
                return $app->redirect($app->url('admin_product_page', array('page_no' => $page_no)).'?resume='.Constant::ENABLED);
0 ignored issues
show
introduced by
Missing blank line before return statement
Loading history...
523
            }
524
525 2
            if ($Product instanceof \Eccube\Entity\Product) {
0 ignored issues
show
Bug introduced by
The class Eccube\Entity\Product does not exist. Did you forget a USE statement, or did you not list all dependencies?

This error could be the result of:

1. Missing dependencies

PHP Analyzer uses your composer.json file (if available) to determine the dependencies of your project and to determine all the available classes and functions. It expects the composer.json to be in the root folder of your repository.

Are you sure this class is defined by one of your dependencies, or did you maybe not list a dependency in either the require or require-dev section?

2. Missing use statement

PHP does not complain about undefined classes in ìnstanceof checks. For example, the following PHP code will work perfectly fine:

if ($x instanceof DoesNotExist) {
    // Do something.
}

If you have not tested against this specific condition, such errors might go unnoticed.

Loading history...
526 2
                log_info('商品削除開始', array($id));
527
528 2
                $Product->setDelFlg(Constant::ENABLED);
529
530 2
                $ProductClasses = $Product->getProductClasses();
531 2
                $deleteImages = array();
532 2
                foreach ($ProductClasses as $ProductClass) {
533 2
                    $ProductClass->setDelFlg(Constant::ENABLED);
534 2
                    $Product->removeProductClass($ProductClass);
535
536 2
                    $ProductClasses = $Product->getProductClasses();
537 2
                    foreach ($ProductClasses as $ProductClass) {
538 2
                        $ProductClass->setDelFlg(Constant::ENABLED);
539 2
                        $Product->removeProductClass($ProductClass);
540
541 2
                        $ProductStock = $ProductClass->getProductStock();
542 2
                        $app['orm.em']->remove($ProductStock);
543
                    }
544
545 2
                    $ProductImages = $Product->getProductImage();
546 2
                    foreach ($ProductImages as $ProductImage) {
547 2
                        $Product->removeProductImage($ProductImage);
548 2
                        $deleteImages[] = $ProductImage->getFileName();
549 2
                        $app['orm.em']->remove($ProductImage);
550
                    }
551
552 2
                    $ProductCategories = $Product->getProductCategories();
553 2
                    foreach ($ProductCategories as $ProductCategory) {
554 2
                        $Product->removeProductCategory($ProductCategory);
555 2
                        $app['orm.em']->remove($ProductCategory);
556
                    }
557
0 ignored issues
show
Coding Style introduced by
Blank line found at end of control structure
Loading history...
558
                }
559
560 2
                $app['orm.em']->persist($Product);
561
562 2
                $app['orm.em']->flush();
563
564 2
                $event = new EventArgs(
565
                    array(
566 2
                        'Product' => $Product,
567 2
                        'ProductClass' => $ProductClasses,
568 2
                        'deleteImages' => $deleteImages,
569
                    ),
570
                    $request
571
                );
572 2
                $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_DELETE_COMPLETE, $event);
573 2
                $deleteImages = $event->getArgument('deleteImages');
574
575
                // 画像ファイルの削除(commit後に削除させる)
576 2
                foreach ($deleteImages as $deleteImage) {
577
                    try {
578 2
                        $fs = new Filesystem();
579 2
                        $fs->remove($app['config']['image_save_realdir'] . '/' . $deleteImage);
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
580 2
                    } catch (\Exception $e) {
581
                        // エラーが発生しても無視する
582
                    }
583
                }
584
585 2
                log_info('商品削除完了', array($id));
586
587 2
                $app->addSuccess('admin.delete.complete', 'admin');
588
            } else {
589
                log_info('商品削除エラー', array($id));
590 2
                $app->addError('admin.delete.failed', 'admin');
591
            }
592
        } else {
593
            log_info('商品削除エラー', array($id));
594
            $app->addError('admin.delete.failed', 'admin');
595
        }
596
597 2
        return $app->redirect($app->url('admin_product_page', array('page_no' => $page_no)).'?resume='.Constant::ENABLED);
598
    }
599
600 2
    public function copy(Application $app, Request $request, $id = null)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
601
    {
602 2
        $this->isTokenValid($app);
603
604 2
        if (!is_null($id)) {
605 2
            $Product = $app['eccube.repository.product']->find($id);
606 2
            if ($Product instanceof \Eccube\Entity\Product) {
0 ignored issues
show
Bug introduced by
The class Eccube\Entity\Product does not exist. Did you forget a USE statement, or did you not list all dependencies?

This error could be the result of:

1. Missing dependencies

PHP Analyzer uses your composer.json file (if available) to determine the dependencies of your project and to determine all the available classes and functions. It expects the composer.json to be in the root folder of your repository.

Are you sure this class is defined by one of your dependencies, or did you maybe not list a dependency in either the require or require-dev section?

2. Missing use statement

PHP does not complain about undefined classes in ìnstanceof checks. For example, the following PHP code will work perfectly fine:

if ($x instanceof DoesNotExist) {
    // Do something.
}

If you have not tested against this specific condition, such errors might go unnoticed.

Loading history...
607 2
                $CopyProduct = clone $Product;
608 2
                $CopyProduct->copy();
609 2
                $Disp = $app['eccube.repository.master.disp']->find(\Eccube\Entity\Master\Disp::DISPLAY_HIDE);
610 2
                $CopyProduct->setStatus($Disp);
611
612 2
                $CopyProductCategories = $CopyProduct->getProductCategories();
613 2
                foreach ($CopyProductCategories as $Category) {
614 2
                    $app['orm.em']->persist($Category);
615
                }
616
617
                // 規格あり商品の場合は, デフォルトの商品規格を取得し登録する.
618 2
                if ($CopyProduct->hasProductClass()) {
619 2
                    $softDeleteFilter = $app['orm.em']->getFilters()->getFilter('soft_delete');
620 2
                    $softDeleteFilter->setExcludes(array(
0 ignored issues
show
introduced by
Add a comma after each item in a multi-line array
Loading history...
621
                        'Eccube\Entity\ProductClass'
622 2
                    ));
623 2
                    $dummyClass = $app['eccube.repository.product_class']->findOneBy(array(
624 2
                        'del_flg' => \Eccube\Common\Constant::ENABLED,
625
                        'ClassCategory1' => null,
626
                        'ClassCategory2' => null,
627 2
                        'Product' => $Product,
628
                    ));
629 2
                    $dummyClass = clone $dummyClass;
630 2
                    $dummyClass->setProduct($CopyProduct);
631 2
                    $CopyProduct->addProductClass($dummyClass);
632 2
                    $softDeleteFilter->setExcludes(array());
633
                }
634
635 2
                $CopyProductClasses = $CopyProduct->getProductClasses();
636 2
                foreach ($CopyProductClasses as $Class) {
637 2
                    $Stock = $Class->getProductStock();
638 2
                    $CopyStock = clone $Stock;
639 2
                    $CopyStock->setProductClass($Class);
640 2
                    $app['orm.em']->persist($CopyStock);
641
642 2
                    $app['orm.em']->persist($Class);
643
                }
644 2
                $Images = $CopyProduct->getProductImage();
645 2
                foreach ($Images as $Image) {
0 ignored issues
show
Coding Style introduced by
Blank line found at start of control structure
Loading history...
646
647
                    // 画像ファイルを新規作成
648 2
                    $extension = pathinfo($Image->getFileName(), PATHINFO_EXTENSION);
649 2
                    $filename = date('mdHis') . uniqid('_') . '.' . $extension;
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
650
                    try {
651 2
                        $fs = new Filesystem();
652 2
                        $fs->copy($app['config']['image_save_realdir'] . '/' . $Image->getFileName(), $app['config']['image_save_realdir'] . '/' . $filename);
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
653 2
                    } catch (\Exception $e) {
654
                        // エラーが発生しても無視する
655
                    }
656 2
                    $Image->setFileName($filename);
657
658 2
                    $app['orm.em']->persist($Image);
659
                }
660 2
                $Tags = $CopyProduct->getProductTag();
661 2
                foreach ($Tags as $Tag) {
662 2
                    $app['orm.em']->persist($Tag);
663
                }
664
665 2
                $app['orm.em']->persist($CopyProduct);
666
667 2
                $app['orm.em']->flush();
668
669 2
                $event = new EventArgs(
670
                    array(
671 2
                        'Product' => $Product,
672 2
                        'CopyProduct' => $CopyProduct,
673 2
                        'CopyProductCategories' => $CopyProductCategories,
674 2
                        'CopyProductClasses' => $CopyProductClasses,
675 2
                        'images' => $Images,
676 2
                        'Tags' => $Tags,
677
                    ),
678
                    $request
679
                );
680 2
                $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_COPY_COMPLETE, $event);
681
682 2
                $app->addSuccess('admin.product.copy.complete', 'admin');
683
684 2
                return $app->redirect($app->url('admin_product_product_edit', array('id' => $CopyProduct->getId())));
685
            } else {
686
                $app->addError('admin.product.copy.failed', 'admin');
687
            }
688
        } else {
689
            $app->addError('admin.product.copy.failed', 'admin');
690
        }
691
692
        return $app->redirect($app->url('admin_product'));
693
    }
694
695 1
    public function display(Application $app, Request $request, $id = null)
0 ignored issues
show
introduced by
Missing function doc comment
Loading history...
696
    {
697 1
        $event = new EventArgs(
698 1
            array(),
699
            $request
700
        );
701 1
        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_DISPLAY_COMPLETE, $event);
702
703 1
        if (!is_null($id)) {
704 1
            return $app->redirect($app->url('product_detail', array('id' => $id, 'admin' => '1')));
705
        }
706
707
        return $app->redirect($app->url('admin_product'));
708
    }
709
710
    /**
711
     * 商品CSVの出力.
712
     *
713
     * @param Application $app
714
     * @param Request $request
0 ignored issues
show
introduced by
Expected 5 spaces after parameter type; 1 found
Loading history...
715
     * @return StreamedResponse
716
     */
717
    public function export(Application $app, Request $request)
718
    {
719
        // タイムアウトを無効にする.
720
        set_time_limit(0);
721
722
        // sql loggerを無効にする.
723
        $em = $app['orm.em'];
724
        $em->getConfiguration()->setSQLLogger(null);
725
726
        $response = new StreamedResponse();
727
        $response->setCallback(function () use ($app, $request) {
728
729
            // CSV種別を元に初期化.
730
            $app['eccube.service.csv.export']->initCsvType(CsvType::CSV_TYPE_PRODUCT);
731
732
            // ヘッダ行の出力.
733
            $app['eccube.service.csv.export']->exportHeader();
734
735
            // 商品データ検索用のクエリビルダを取得.
736
            $qb = $app['eccube.service.csv.export']
737
                ->getProductQueryBuilder($request);
738
739
            // joinする場合はiterateが使えないため, select句をdistinctする.
740
            // http://qiita.com/suin/items/2b1e98105fa3ef89beb7
741
            // distinctのmysqlとpgsqlの挙動をあわせる.
742
            // http://uedatakeshi.blogspot.jp/2010/04/distinct-oeder-by-postgresmysql.html
743
            $qb->resetDQLPart('select')
744
                ->resetDQLPart('orderBy')
745
                ->select('p')
746
                ->orderBy('p.update_date', 'DESC')
747
                ->distinct();
748
749
            // データ行の出力.
750
            $app['eccube.service.csv.export']->setExportQueryBuilder($qb);
751 View Code Duplication
            $app['eccube.service.csv.export']->exportData(function ($entity, $csvService) use ($app, $request) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated across your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
752
753
                $Csvs = $csvService->getCsvs();
754
755
                /** @var $Product \Eccube\Entity\Product */
756
                $Product = $entity;
757
758
                /** @var $Product \Eccube\Entity\ProductClass[] */
759
                $ProductClassess = $Product->getProductClasses();
0 ignored issues
show
Bug introduced by
The method getProductClasses cannot be called on $Product (of type array<integer,object<Eccube\Entity\ProductClass>>).

Methods can only be called on objects. This check looks for methods being called on variables that have been inferred to never be objects.

Loading history...
760
761
                foreach ($ProductClassess as $ProductClass) {
762
                    $ExportCsvRow = new \Eccube\Entity\ExportCsvRow();
763
764
                    // CSV出力項目と合致するデータを取得.
765
                    foreach ($Csvs as $Csv) {
766
                        // 商品データを検索.
767
                        $ExportCsvRow->setData($csvService->getData($Csv, $Product));
768
                        if ($ExportCsvRow->isDataNull()) {
769
                            // 商品規格情報を検索.
770
                            $ExportCsvRow->setData($csvService->getData($Csv, $ProductClass));
771
                        }
772
773
                        $event = new EventArgs(
774
                            array(
775
                                'csvService' => $csvService,
776
                                'Csv' => $Csv,
777
                                'ProductClass' => $ProductClass,
778
                                'ExportCsvRow' => $ExportCsvRow,
779
                            ),
780
                            $request
781
                        );
782
                        $app['eccube.event.dispatcher']->dispatch(EccubeEvents::ADMIN_PRODUCT_CSV_EXPORT, $event);
783
784
                        $ExportCsvRow->pushData();
785
                    }
786
787
                    //$row[] = number_format(memory_get_usage(true));
0 ignored issues
show
Unused Code Comprehensibility introduced by
65% of this comment could be valid code. Did you maybe forget this after debugging?

Sometimes obsolete code just ends up commented out instead of removed. In this case it is better to remove the code once you have checked you do not need it.

The code might also have been commented out for debugging purposes. In this case it is vital that someone uncomments it again or your project may behave in very unexpected ways in production.

This check looks for comments that seem to be mostly valid code and reports them.

Loading history...
788
                    // 出力.
789
                    $csvService->fputcsv($ExportCsvRow->getRow());
790
                }
791
            });
792
        });
793
794
        $now = new \DateTime();
795
        $filename = 'product_' . $now->format('YmdHis') . '.csv';
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
796
        $response->headers->set('Content-Type', 'application/octet-stream');
797
        $response->headers->set('Content-Disposition', 'attachment; filename=' . $filename);
0 ignored issues
show
Coding Style introduced by
Concat operator must not be surrounded by spaces
Loading history...
798
        $response->send();
799
800
        log_info('商品CSV出力ファイル名', array($filename));
801
802
        return $response;
803
    }
804
    
0 ignored issues
show
introduced by
Please trim any trailing whitespace
Loading history...
805
    /**
806
     * ProductCategory作成
807
     * @param \Eccube\Entity\Product $Product
808
     * @param \Eccube\Entity\Category $Category
809
     * @return \Eccube\Entity\ProductCategory
810
     */
811 View Code Duplication
    private function createProductCategory($Product, $Category, $count)
0 ignored issues
show
Duplication introduced by
This method seems to be duplicated in your project.

Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.

You can also find more detailed suggestions in the “Code” section of your repository.

Loading history...
812
    {
813
        $ProductCategory = new \Eccube\Entity\ProductCategory();
814
        $ProductCategory->setProduct($Product);
815
        $ProductCategory->setProductId($Product->getId());
816
        $ProductCategory->setCategory($Category);
817
        $ProductCategory->setCategoryId($Category->getId());
818
        $ProductCategory->setRank($count);
819
        
0 ignored issues
show
introduced by
Please trim any trailing whitespace
Loading history...
820
        return $ProductCategory;
821
    }
822
}
823