web-dev-qa-db-ja.com

SharePoint2010-クライアントオブジェクトモデル-ListItemに添付ファイルを追加します

クライアントオブジェクトモデルを使用して新しいListItemを追加するSharePointリストがあります。 ListItemsの追加は問題ではなく、うまく機能します。

次に、添付ファイルを追加します。

私は次の方法でSaveBinaryDirectを使用しています。

File.SaveBinaryDirect(clientCtx, url.AbsolutePath + "/Attachments/31/" + fileName, inputStream, true);

添付ファイルを追加しようとしているアイテムに、クライアントオブジェクトモデルを使用せずにSharePointサイトから追加された添付ファイルが既に含まれている限り、問題なく機能します。

まだ添付ファイルがないアイテムに添付ファイルを追加しようとすると、次のエラーが発生します(両方とも発生しますが、同じファイルでは発生しませんが、これら2つのメッセージは一貫して表示されます)。

リモートサーバーがエラーを返しました:(409)競合
リモートサーバーがエラーを返しました:(404)見つかりません

このアイテムの添付フォルダを最初に作成する必要があるかもしれないと思いました。次のコードを試してみると:

clientCtx.Load(ticketList.RootFolder.Folders);
clientCtx.ExecuteQuery();
clientCtx.Load(ticketList.RootFolder.Folders[1]);             // 1 -> Attachment folder
clientCtx.Load(ticketList.RootFolder.Folders[1].Folders);
clientCtx.ExecuteQuery();
Folder folder = ticketList.RootFolder.Folders[1].Folders.Add("33");
clientCtx.ExecuteQuery();

次のようなエラーメッセージが表示されます。

「リスト/チケットシステム/添付ファイル/ 33」フォルダを作成できません

SharePointサイト/リストの完全な管理者権限を持っています。

私が間違っている可能性のあるアイデアはありますか?

ありがとう、ソーベン

14
Thorben

この質問についてマイクロソフトと話し合いました。添付ファイルをリモートで作成する唯一の方法は、List.asmxWebサービスのようです。私もこのサブフォルダーを作成しようとしましたが、成功しませんでした。

9
MaxBeard

私もこの問題に長い間苦労していたので、リストアイテムを正常に作成して添付ファイルを追加する方法を示す完全なコードサンプルを投稿したいと思いました。

クライアントオブジェクトAPIを使用してリストアイテムを作成し、SOAP Webサービスを使用して添付ファイルを追加しています。これは、Webの他の場所で説明されているように、クライアントオブジェクトAPIがアイテムのアップロードディレクトリがすでに存在するアイテムに添付ファイルを追加するためにのみ使用されます(たとえば、アイテムにすでに添付ファイルがある場合)。それ以外の場合は、409エラーなどで失敗します。SOAP webただし、サービスはこれに対応します。

私が克服しなければならなかったもう1つのことは、次のURLを使用してSOAP参照を追加したにもかかわらず、

https://my.sharepoint.installation/personal/test/_vti_bin/lists.asmx

VSが実際にapp.configに追加したURLは次のとおりです。

https://my.sharepoint.installation/_vti_bin/lists.asmx

App.configを手動で正しいURLに戻す必要がありました。そうしないと、エラーが発生します。

リストが存在しません。選択したページに、存在しないリストが含まれています。別のユーザーによって削除された可能性があります。 0x82000006

コードは次のとおりです。

    void CreateWithAttachment()
    {
        const string listName = "MyListName";
        // set up our credentials
        var credentials = new NetworkCredential("username", "password", "domain");

        // create a soap client
        var soapClient = new ListsService.Lists();
        soapClient.Credentials = credentials;

        // create a client context
        var clientContext = new Microsoft.SharePoint.Client.ClientContext("https://my.sharepoint.installation/personal/test");
        clientContext.Credentials = credentials;

        // create a list item
        var list = clientContext.Web.Lists.GetByTitle(listName);
        var itemCreateInfo = new ListItemCreationInformation();
        var newItem = list.AddItem(itemCreateInfo);

        // set its properties
        newItem["Title"] = "Created from Client API";
        newItem["Status"] = "New";
        newItem["_Comments"] = "here are some comments!!";

        // commit it
        newItem.Update();
        clientContext.ExecuteQuery();

        // load back the created item so its ID field is available for use below
        clientContext.Load(newItem);
        clientContext.ExecuteQuery();

        // use the soap client to add the attachment
        const string path = @"c:\temp\test.txt";
        soapClient.AddAttachment(listName, newItem["ID"].ToString(), Path.GetFileName(path),
                                  System.IO.File.ReadAllBytes(path));
    }

これが誰かを助けることを願っています。

12

Sharepoint 2010では、COMを使用してリストアイテムに最初の添付ファイルをアップロードする方法はありませんでした。代わりにListsWebサービスを使用することをお勧めします。

Sharepoint2013では機能します。

AttachmentCreationInformation newAtt = new AttachmentCreationInformation();
newAtt.FileName = "myAttachment.txt";
// create a file stream
string fileContent = "This file is was ubloaded by client object meodel ";
System.Text.ASCIIEncoding enc = new System.Text.ASCIIEncoding();
byte[] buffer = enc.GetBytes(fileContent);
newAtt.ContentStream = new MemoryStream(buffer);

// att new item or get existing one
ListItem itm = list.GetItemById(itemId);
ctx.Load(itm);   
// do not execute query, otherwise a "version conflict" exception is rised, but the file         is uploaded
// add file to attachment collection
newAtt.ContentStream = new MemoryStream(buffer);
itm.AttachmentFiles.Add(newAtt); 
AttachmentCollection attachments = itm.AttachmentFiles;
ctx.Load(attachments);
ctx.ExecuteQuery(); 
// see all attachments for list item
// this snippet works if the list item has no attachments

このメソッドは http://www.mailtosharepoint.net/ で使用されます

6
sam

これは、問題の承認とその解決方法に関する有用な提案を提出しなかったため、MicrosoftSharePointチームにはあまり反映されていません。これが私がそれをどのように扱ったかです:

製品に付属している新しいSharePoint2010管理対象クライアントを使用しています。したがって、私はすでに資格情報を持つSharePointClientContextを持っています。次の関数は、リストアイテムに添付ファイルを追加します。

private void SharePoint2010AddAttachment(ClientContext ctx, 
                                     string listName, string itemId, 
                                     string fileName, byte[] fileContent)
{
    var listsSvc = new sp2010.Lists();
    listsSvc.Credentials = _sharePointCtx.Credentials;
    listsSvc.Url = _sharePointCtx.Web.Context.Url + "_vti_bin/Lists.asmx";
    listsSvc.AddAttachment(listName, itemId, fileName, fileContent);
}

上記のコードの唯一の前提条件は、プロジェクト(Visual Studio 2008を使用)に、次のURLから作成されたsp2010という_web_reference_を追加することです。http:/// _ vti_bin/Lists.asmx

ボンチャンス...

4
Dror Harari

これをCSOM(SharePointクライアントオブジェクトモデル)アプリケーションで使用して試しましたが、機能します

using (ClientContext context = new ClientContext("http://spsite2010"))
                {

                    context.Credentials = new NetworkCredential("admin", "password");
                    Web oWeb = context.Web;
                    List list = context.Web.Lists.GetByTitle("Tasks");
                    CamlQuery query = new CamlQuery();
                    query.ViewXml = "<View><Where><Eq><FieldRef Name = \"Title\"/><Value Type=\"String\">New Task Created</Value></Eq></Where></View>";
                    ListItemCollection listItems = list.GetItems(query);
                    context.Load(listItems);
                    context.ExecuteQuery();
                    FileStream oFileStream = new FileStream(@"C:\\sample.txt", FileMode.Open);
                    string attachmentpath = "/Lists/Tasks/Attachments/" + listItems[listItems.Count - 1].Id + "/sample.txt";
                    Microsoft.SharePoint.Client.File.SaveBinaryDirect(context, attachmentpath, oFileStream, true);
                }

注:アイテムフォルダがすでに作成されている場合にのみ機能します

0
kmanxi

HTML:

<asp:FileUpload ID="FileUpload1" runat="server" AllowMultiple="true" />

コードビハインドのイベント:

protected void UploadMultipleFiles(object sender, EventArgs e)
{
    Common.UploadDocuments(Common.getContext(new Uri(Request.QueryString["SPHostUrl"]),
    Request.LogonUserIdentity), FileUpload1.PostedFiles, new CustomerRequirement(), 5);
}

public static List<string> UploadDocuments<T>(ClientContext ctx,IList<HttpPostedFile> selectedFiles, T reqObj, int itemID)
{
    List<Attachment> existingFiles = null;
    List<string> processedFiles = null;
    List<string> unProcessedFiles = null;
    ListItem item = null;
    FileStream sr = null;
    AttachmentCollection attachments = null;
    byte[] contents = null;
    try
    {
        existingFiles = new List<Attachment>();
        processedFiles = new List<string>();
        unProcessedFiles = new List<string>();
        //Get the existing item
        item = ctx.Web.Lists.GetByTitle(typeof(T).Name).GetItemById(itemID);
        //get the Existing attached attachments
        attachments = item.AttachmentFiles;
        ctx.Load(attachments);
        ctx.ExecuteQuery();
        //adding into the new List
        foreach (Attachment att in attachments)
            existingFiles.Add(att);
        //For each Files which user has selected
        foreach (HttpPostedFile postedFile in selectedFiles)
        {
            string fileName = Path.GetFileName(postedFile.FileName);
            //If selected file not exist in existing item attachment
            if (!existingFiles.Any(x => x.FileName == fileName))
            {
                //Added to Process List
                processedFiles.Add(postedFile.FileName);
            }
            else
                unProcessedFiles.Add(fileName);
        }
        //Foreach process item add it as an attachment
        foreach (string path in processedFiles)
        {
            sr = new FileStream(path, FileMode.Open);
            contents = new byte[sr.Length];
            sr.Read(contents, 0, (int)sr.Length);
            var attInfo = new AttachmentCreationInformation();
            attInfo.FileName = Path.GetFileName(path);
            attInfo.ContentStream = sr;
            item.AttachmentFiles.Add(attInfo);
            item.Update();
        }
        ctx.ExecuteQuery();
    }
    catch (Exception ex)
    {
        throw ex;
    }
    finally
    {
        existingFiles = null;
        processedFiles = null;
        item = null;
        sr = null;
        attachments = null;
        contents = null;
        ctx = null;

    }
    return unProcessedFiles;
}
0
Jaydeep Shil