通常は権限のないユーザーからシンボリックリンクを介してPHPのファイルにアクセスする

通常は権限のないユーザーからシンボリックリンクを介してPHPのファイルにアクセスする

私にはこの Web サイトがあります。ユーザーがフォームを送信すると、Python スクリプトが PHP ページを通じて実行され、Python スクリプトによって zip ファイルが作成され、リンクを通じてダウンロードできるようにユーザーに提供されます。ファイルは巨大になる可能性があります (数 GB)。

私は大学のサーバーで作業しているので、そのサーバーのルールと機能に厳密に従わなければなりません。問題は次のとおりです。

この Web サイトは に保存されています/data/mywebsiteが、ディスク容量が限られています。もちろん、この Webサイトwww-dataは Apache サーバーから主にアクセスできるため、 が所有しています。

に 1 TB のストレージが提供されていますが/experimentdata/、これは という特定の 1 人のユーザーのみがアクセスできますtheuser。これは、このフォルダーが 1 人の特定のユーザー ID でアクセスできる samba マウントであるためです。

にファイルを作成するには、ユーザー としてファイルを作成するコマンドを/experimentdata使用します。ここで問題となるのは、このファイルを Apache 経由でダウンロードするためのリンクで提供するにはどうすればよいですか?sudo -u theuser/experimentdata/downloadme.ziptheuser

たとえば、 に置いたシンボリック リンクを使用することを考えました/data/mywebsite/download/downloadme.zip。その問題は、ユーザーにwww-dataファイルを読み取る権限がまったくないことです。

/experimentdata/downloadme.zipユーザーがユーザーをwww-data介してファイルをダウンロードできるようにするにはどうすればよいでしょうかtheuser?

関与することはまったく問題ないと明確に言いたいのですsudo -u theuserが、そこから自分の Web サイト フォルダー外のどこかへのリンクを作成する方法がわかりません。

追伸: 追加情報が必要な場合はお問い合わせください。

答え1

やるべきことは、 の代わりにphp/ がpython直接データを返すようにすることだと思いますapache。コードでは、 が行うのと同じことを行うことができますapache。私の経験では、これは別のディレクトリを開いたり、 を使用したりsudo、 のファイル権限を変更したりするよりもはるかに優れていますapache

プログラムがインターネット接続よりも速く大きなファイルを生成する場合は、プログラムから直接データをストリーミングできます。これにより、余分なデータ ファイルとそれを管理するためのコード、およびそれを記憶するためのメカニズムが不要になります。

Stack Overflow のこの回答では、コードが でどのように機能するかを示していますphphttps://stackoverflow.com/a/4357904/5484716

このように呼び出されるプログラムの場合、すべてのstderrストリーム出力を削除し、Python プロセスからの戻りコードがプロセスの成功または失敗を正確に反映していることを確認します。

以下の例は、popen()stackoverflow の上記の例のシナリオで使用する呼び出しを示しています。 をシェル コマンドの前に追加しました。これにより、 と の両方にデータが来るとでデッドロックの原因になる可能性があるexec 2>/dev/null;ため、シェル自体からでも標準エラーに出力が送られないことが保証されます。stderrstdoutpopen()

ディスク ファイルをユーザーにダウンロードする場合:

$fp = popen('exec 2>/dev/null; sudo -u theuser cat yourfile.zip', 'r');

アクティブなプロセスからデータをダウンロードする場合:

$fp = popen('exec 2>/dev/null; sudo yourpythonscript arg argN', 'r');

これらのコマンドラインシェルコマンドであり、シェルのメタ文字に対して適切に引用符で囲む必要があります。

2 番目の方法では、サーバーはすぐにデータの送信を開始します。ユーザーがフォームを正常に送信すると、すぐにブラウザに「名前を付けて保存」ダイアログが表示されます。ユーザーが出力ファイルを選択すると、phpスクリプトはデータを直接ネットワーク経由でリモート ファイルに転送します。

スクリプトはpython次のように印刷されますのみ標準出力に zip データを出力し、 zip プロセスの成功または失敗を正確に表す終了コードを返します。pythonスクリプトではsys.stdout、たとえばに出力を書き込む必要がありますzf = ZipFile(sys.stdout, ...

pclose()呼び出して戻り値を確認することは重要ですなぜなら、それが zip が成功したかどうかを知る唯一の方法だからです。 がpclose()0 以外を返す場合、何かが間違っています。

クライアントによるファイルの処理方法は、以下の設定response headersなどによって異なります: content-type:content-encoding:content-disposition: 参照:http://www.w3.org/Protocols/rfc2616/rfc2616-sec6.htmlresponse-headerおよびentity-header情報を参照してください。

関連情報