6

アップロードしたファイルをネットワーク共有に保存するASP.NETプロジェクトに取り組んでいます。仮想ディレクトリを使用すれば問題ないと思いましたが、Directory.CreateDirectoryのアクセス許可に苦労しています。

ファイルをアップロードできたので、コードを変更してすべてを1つのディレクトリに配置することにしましたが、重複を上書きしないようにFile.Existsを使用する必要があります。

すべてのコードを更新したので、何をしても、ネットワーク共有に対してテストすると、File.Existsは常にfalse(ファイルは間違いなく存在します)を返すことがわかりました。

何か案は?私はネットワーク共有で私のロープの最後に来ています。

4

4 に答える 4

8

File.Existは、実際にはファイルの存在をチェックしません。代わりに、ある程度アクセスできるファイルの存在をチェックします。ファイルが存在することがわかっている場合は、ファイルにアクセスできないことが問題である可能性があります。

于 2008-12-30T19:58:08.477 に答える
7

私は最近、ファイルをネットワーク共有に保存する非常によく似たプロジェクトに取り組みました。2台のコンピューターは同じサブネット上にありますが、ドメインコントローラーによって制御されていないため、各コンピューターには独自のユーザーがいます。

両方のコンピューターで同じユーザー名とパスワードのユーザーを作成しました。次に、ネットワーク共有を作成し、ユーザーが読み取り/書き込みできるようにフォルダー/共有のアクセス許可を設定しました。

次に、なりすましを管理するために次のクラスを作成しました。

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using System.Security.Principal;
using System.Security.Permissions;
using System.Text;

namespace MyProject.Business.Web
{
    public class SecurityManager
    {
        #region DLL Imports
        [DllImport("advapi32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
        public static extern bool LogonUser(String lpszUsername, String lpszDomain, String lpszPassword, int dwLogonType, int dwLogonProvider, ref IntPtr phToken);

        [DllImport("kernel32.dll", CharSet = CharSet.Auto)]
        public extern static bool CloseHandle(IntPtr handle);

        [DllImport("advapi32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        public extern static bool DuplicateToken(IntPtr ExistingTokenHandle, int SECURITY_IMPERSONATION_LEVEL, ref IntPtr DuplicateTokenHandle);
        #endregion

        public string Domain { get; set; }
        public string UserName { get; set; }
        public string Password { get; set; }

        private WindowsImpersonationContext m_CurrentImpersonationContext;

        [PermissionSetAttribute(SecurityAction.Demand, Name = "FullTrust")]
        public void StartImpersonation()
        {
            const int LOGON32_PROVIDER_DEFAULT = 0;
            const int LOGON32_LOGON_INTERACTIVE = 2;

            IntPtr tokenHandle = IntPtr.Zero;
            IntPtr dupeTokenHandle = IntPtr.Zero;

            // obtain a handle to an access token
            bool wasLogonSuccessful = LogonUser(UserName, Domain, Password, LOGON32_LOGON_INTERACTIVE, LOGON32_PROVIDER_DEFAULT, ref tokenHandle);

            if (!wasLogonSuccessful)
                throw new Exception(String.Format("Logon failed with error number {0}", Marshal.GetLastWin32Error()));

            // use the token handle to impersonate the user
            WindowsIdentity newId = new WindowsIdentity(tokenHandle);
            m_CurrentImpersonationContext = newId.Impersonate();

            // free the tokens
            if (tokenHandle != IntPtr.Zero)
                CloseHandle(tokenHandle);
        }
        public void EndImpersonation()
        {
            m_CurrentImpersonationContext.Undo();
        }
    }
}

次に、ASP.NETページで次のことを行いました。

SecurityManager sm = new SecurityManager();
sm.UserName = ConfigurationManager.AppSettings["UserFileShareUsername"];
sm.Password = ConfigurationManager.AppSettings["UserFileSharePassword"];
sm.StartImpersonation();

if (!Directory.Exists(folderPath)) Directory.CreateDirectory(folderPath);

File.Move(sourcePath, destinationPath);

sm.EndImpersonation();
于 2008-12-30T20:03:17.653 に答える
2

おそらく、実行中のコード(つまり、ASP.NETサーバーコード)は、そのネットワーク共有にアクセスするためのアクセス許可を持たないユーザー(IISユーザーなど)として実行されています。

IISは、他のマシンで共有を表示する権限をデフォルトで持っている、高度な特権を持つユーザーとして実行されることは想定されていないと思います。

于 2008-12-30T19:54:11.260 に答える
2

ほとんど同じコードを使用しましたが、クラスに IDisposable インターフェイスを実装させ、Dispose() メソッドに Undo() を追加しました。あなたがこのコードを使用する唯一の開発者であり、常に正しい方法で作業を行う場合、このコードは正常に機能しますよね?

于 2010-06-28T15:01:41.920 に答える