回答編集履歴

4

サンプルコードを編集

2017/05/07 17:48

投稿

shiroyuki
shiroyuki

スコア169

test CHANGED
@@ -1,3 +1,7 @@
1
+ Warningを見たところ$_GET['file']が取得できていないようですね。
2
+
3
+
4
+
1
5
  まずgetのパラメータにurlを構成する文字列等が入る場合はurlエンコードしたものを書く必要があります。
2
6
 
3
7
  http://~~/download.php?file=audio%2fsample%2emp3
@@ -20,6 +24,10 @@
20
24
 
21
25
 
22
26
 
27
+ 複雑なサイトでなければ配列を利用した簡易チェックが楽でしょうか。
28
+
29
+ アクセスするURL http://~~/download.php?file=sample1
30
+
23
31
  ```php
24
32
 
25
33
  <?php
@@ -34,11 +42,37 @@
34
42
 
35
43
 
36
44
 
45
+ // この配列に含まれないものはダウンロードできないようにする
46
+
47
+ $CONTENTS_LIST = [
48
+
49
+ // キー名にgetで渡す値、要素にファイルパス
50
+
37
- $file = $_GET['file'];
51
+ 'sample1' => 'audio/sample.mp3',
52
+
53
+ 'sample2' => 'audio/sample2.mp3',
54
+
55
+ 'sample3' => 'audio/sample3.mp3',
56
+
57
+ 'sample4' => 'audio/sample4.mp3',
58
+
59
+ ];
38
60
 
39
61
 
40
62
 
41
- // ファイルの存在チェックや参照してはいけないファイルのチェックなどはこのあたりに書
63
+ // $CONTENTS_LISTに含まれないファイルを弾
64
+
65
+ if ( !in_array( $_GET['file'], array_keys($CONTENTS_LIST) ) ) {
66
+
67
+ echo 'ファイル名が異常です';
68
+
69
+ exit;
70
+
71
+ }
72
+
73
+
74
+
75
+ $file = $CONTENTS_LIST[ $_GET['file'] ];
42
76
 
43
77
  if ( !file_exists( $file ) ) {
44
78
 
@@ -47,8 +81,6 @@
47
81
  exit;
48
82
 
49
83
  }
50
-
51
-
52
84
 
53
85
  // ファイルを出力
54
86
 

3

サンプルコード追記

2017/05/07 17:48

投稿

shiroyuki
shiroyuki

スコア169

test CHANGED
@@ -1,15 +1,63 @@
1
- getのパラメータにurlを構成する文字列等が入る場合はurlエンコードしたものを書く必要があります。
1
+ まずgetのパラメータにurlを構成する文字列等が入る場合はurlエンコードしたものを書く必要があります。
2
2
 
3
3
  http://~~/download.php?file=audio%2fsample%2emp3
4
4
 
5
5
 
6
6
 
7
- あとheader出力前のprintは実際のダウンロード時は消しましょう。
7
+ あとテストコードと思われますが、header出力前のprintは実際のダウンロード時は消しましょう。```print("$file<br>\n");```
8
8
 
9
+
10
+
9
- ```print("$file<br>\n");```
11
+ [PHP: header - Manual](https://secure.php.net/manual/ja/function.header.php)
12
+
13
+ > header() 関数は、 通常の HTML タグまたは PHP からの出力にかかわらず、すべての実際の 出力の前にコールする必要があることです。
10
14
 
11
15
 
12
16
 
13
17
  これは別件ですがgetで渡された値をノーチェックでreadfileに渡すのはかなり恐ろしいことなので
14
18
 
15
19
  外部に公開するようなページの場合は参照してはいけないファイルではないかのチェックは行ったほうがよいでしょう。
20
+
21
+
22
+
23
+ ```php
24
+
25
+ <?php
26
+
27
+ if ( empty( $_GET['file'] ) {
28
+
29
+ echo 'ファイル名が取得できませんでした';
30
+
31
+ exit;
32
+
33
+ }
34
+
35
+
36
+
37
+ $file = $_GET['file'];
38
+
39
+
40
+
41
+ // ファイルの存在チェックや参照してはいけないファイルのチェックなどはこのあたりに書く
42
+
43
+ if ( !file_exists( $file ) ) {
44
+
45
+ echo '存在しないファイルです';
46
+
47
+ exit;
48
+
49
+ }
50
+
51
+
52
+
53
+ // ファイルを出力
54
+
55
+ header('Content-Type: application/force-download');
56
+
57
+ header('Content-Length: '.filesize($file));
58
+
59
+ header('Content-Disposition: attachment; filename="'.$file.'"');
60
+
61
+ readfile($file);
62
+
63
+ ```

2

編集

2017/05/07 17:25

投稿

shiroyuki
shiroyuki

スコア169

test CHANGED
@@ -4,6 +4,12 @@
4
4
 
5
5
 
6
6
 
7
+ あとheader出力前のprintは実際のダウンロード時は消しましょう。
8
+
9
+ ```print("$file<br>\n");```
10
+
11
+
12
+
7
13
  これは別件ですがgetで渡された値をノーチェックでreadfileに渡すのはかなり恐ろしいことなので
8
14
 
9
15
  外部に公開するようなページの場合は参照してはいけないファイルではないかのチェックは行ったほうがよいでしょう。

1

セキュリティ面での注意書き追加

2017/05/07 16:31

投稿

shiroyuki
shiroyuki

スコア169

test CHANGED
@@ -1,3 +1,9 @@
1
1
  getのパラメータにurlを構成する文字列等が入る場合はurlエンコードしたものを書く必要があります。
2
2
 
3
3
  http://~~/download.php?file=audio%2fsample%2emp3
4
+
5
+
6
+
7
+ これは別件ですがgetで渡された値をノーチェックでreadfileに渡すのはかなり恐ろしいことなので
8
+
9
+ 外部に公開するようなページの場合は参照してはいけないファイルではないかのチェックは行ったほうがよいでしょう。