понедельник, 22 июля 2013 г.

Разное

Передача параметров в вызываемую форму:
http://axforum.info/forums/showthread.php?t=30177
http://www.axaptapedia.com/Passing_values_between_forms

Фильтрация на форме:
http://dynamics-ax-live.blogspot.ru/2010/03/how-to-filter-records-in-form-by-code.html

Раскраска грида:
http://daxguy.blogspot.ru/2007/04/coloring-grids-in-dax.html

Деактивация контрола (на active датасорса):
    Control.enabled(true);
    select count(RecId) from cSST where cSST.Row == Table_ds.cursor().getFieldValue("RecId");
    if(cSST.RecId <= 1) Control.enabled(false);

DateTime2Real:
public real DT2Real(utcDateTime _dt)
{
    str qwe;
    int wer;
    real ert;  
   
       qwe = DateTimeUtil::toStr(_dt);  
       qwe = strRem(qwe,"T");
       qwe = strRem(qwe,"-");
       qwe = strRem(qwe,":");
       wer = str2int(qwe);
       ert = wer/10000;    
       return ert;
}

Date2UTCDateTime:
public utcDateTime convertDToUtcDT(date simpleDate)
{
int timeOfDay;
UTCDateTime datetime;
;
timeOfDay = str2time('00:00:00');
datetime = DateTimeUtil::newDateTime(simpleDate, timeOfDay);
return datetime;
}

понедельник, 27 мая 2013 г.

Выполнение X++ кода в CLR - код может выполняться быстрее. Раза в три(!) быстрее...

Очень полезная статья, следуя которой можно запилить выполнение клиентского кода в CIL: http://daxmusings.codecrib.com/2011/08/x-in-net-clr-life-in-fast-lane.html

и при необходимости приаттачиться к процессу отладчиком VisualStudio

Вот здесь http://blogs.msdn.com/b/mfp/archive/2012/06/25/the-compare-tool-and-running-x-code-as-il.aspx есть небольшое сравнительное исследование быстродействия.

понедельник, 29 апреля 2013 г.

Создание master/detail формы

Часто возникающая задача - в верхней (левой) части формы список header'ов, а в нижней (или правой) части - список details'ов. Здесь популярно расписано решение: http://daxdave.blogspot.ru/2006/10/creating-headerdetail-based-linkage-on.html
PS да, и тут ни строчки кода ;)
 PPS  и вдогонку оставлю тут ссылочку про как сделать вертикальный сплиттер на форме:  http://dynamicsaxposed.wordpress.com/2011/08/17/how-to-make-vertical-splitter-on-forms-in-ax-2012/

среда, 24 апреля 2013 г.

Зависимые Lookup на форме

Возникла необходимость сделать серию из трёх зависимых полей, выбор значения в одном из них фильтрует значения во втором, выбор второго фильтрует третий. Азбучный пример из книги (перекрытие метода lookup) отлично подходит для поля StringEdit, но отказывается работать на ReferenceGroup. Для последнего необходимо перекрывать lookupReference.
public void lookup() //для StringEdit

воскресенье, 7 апреля 2013 г.

“Продвинутый” фильтр на форме

Microsoft Dynamics AX 2012 for Developers [AX 2012]
Форма позволяет фильтровать Grid различными способами, но нам захотелось сделать свой – по нажатию кнопки открывать дочернюю форму, в которой задать определенные поля, которые сформируют нужный Query.
Первая попытка – сделать extends RunBase класс, в котором создавать DialogField. Всё было б ничего, но не получилось следующее – хотелось вывести в выпадающем меню фильтра Name, а передать в query родительской формы RecId.
Вторая попытка – вызывать из формы всё тот же класс, сделать от этого класса форму, в которой можно разместить ReferenceField, который реализует нужный нам функционал…
Оба эти примера не доделаны, сюда скинул промежуточный итог.
Рабочий и самый простой вариант – одна простая форма. Тут её код:

public class FormRun extends ObjectRun
{
    Object caller;
    boolean activate;
}
void closeOk()
{
    super();
    caller = element.args().caller();
    caller.itemRangeValue(itemIdStringEdit.text());
    caller.subitemNumberValue(subitemNumberIntEdit.value());
    caller.subitemGroupValue(subitemGroupStringEdit.text());
    caller.subitemgoValue(subitemgoReferenceGroup.value());
    if (subitemgoYesNo.checked()) {
        caller.subitemgoYesNo(true);
    }
    caller.1stplaceValue(1stplaceReferenceGroup.value());
    caller.currentplaceValue(CurrentplaceReferenceGroup.value());
    caller.activateFilter(true);
    caller.update();
   ));
}

и код формы – caller’a:

SSRS Отчеты в Dynamics AX 2012

Developing Reports for Microsoft Dynamics AX in Visual Studio [AX 2012]

Очень полезный фрагмент книжки о разработке отчетов – позволяет быстро начать, даже если раньше вам не приходилось работать с Reporting Services + Dynamics.
Находится тут: http://www.slideshare.net/harshnmh/developing-ssrsreportsfordynamicsax
+для себя копия
Задача стояла следующая – необходимо разработать простой отчет, в котором на входе должен быть параметр, взятый из грида. На форме размещается кнопка, по нажатию на кнопку Action – class, код класса ниже:

class RdpReport extends SrsReportRunController
{
    #define.ReportName('QueryBasedReport.Report')
}
protected void prePromptModifyContract()
{
    this.setRanges(this.parmReportContract().parmQueryContracts().lookup(this.getFirstQueryContractKey()));
}
public static client void main(Args _args)
{    rdpReport controller = new rdpReport();
    controller.parmReportName(#ReportName);
    controller.parmArgs(_args);
    controller.startOperation();
}
private void setRanges(Query _query)
{
    real volumeRange;
    Report ReportLocal;
  
    if (this.parmArgs())
    {
                ReportLocal = this.parmArgs().record();
                volumeRange = ReportLocal.Volume;
    }
   
    if (volumeRange)
    {
        SysQuery::findOrCreateRange(
            _query.dataSourceTable(tableNum(Report)),
            fieldNum(Report, Volume)).value(int2str(volumeRange));
    }   
}

среда, 30 января 2013 г.

Получение данных из внешних БД


Рассмотрим пару вариантов работы с внешними БД.

Вариант 1: отобразить как список таблицу из внешней БД MS SQL, позволив пользователям работать с таблицей как со списком – просматривать, добавлять и удалять значения – строки.

Для этого нужно создать внешний тип контента в SP Designer:
ex1
ex2ex3

ex4
Выбрав “Подключиться с удостоверением пользователя”, мы сможем подключиться только к БД, расположенной на том же сервере, где находится БД SharePoint. При попытке подключения к какой-либо другой, получим ошибку, описанную вот тут:   http://www.spdoctor.net/Pages/message.aspx?name=login-failed-for-user-bdc
Как правило, подключение требуется к внешней БД, поэтому надо использовать 2й и 3ий варианты подключения. Эти варианты требуют идентификатора Secure Store – фактически это логин и пароль, в зашифрованном виде хранящиеся в специальной службе SharePoint. Создать Secure Store ID можно через “Центр Администрирования SharePoint” – “Управление приложениями-службами“. Там нужно найти “Приложение службы Secure Store” – называться оно может как угодно.
ВАЖНО: если таких служб 2 или более, обязательно выясните, какая из них используется веб-приложением “по умолчанию”. Сделать это можно здесь: “Центр администрирования - Сопоставления приложений-служб “. Именно её и нужно использовать для создания Secure Store ID.

ex5
По клику на “Создать” вызывается диалог, в котором создаются поля логина и пароля, затем можно задать эти логин и пароль, кликнув “Настроить” в блоке “Учётные данные”
Всё, SSID готов. Теперь можно указать его в поле “Идентификатор приложения SecureStore” и нам станут доступны таблицы внешнего источника данных. Для любой из них можно создать 5 типов операций:
ex6
Для того, чтобы отобразить содержимое таблицы как список – необходимо создать как минимум первые две операции. В создании нет ничего сложного – просто жмём Далее и Финиш.
Ещё один важный момент, без которого список не будет доступен – необходимо разрешить пользователям доступ к этому типу контента – это тоже делается в
“Центр Администрирования SharePoint” – “Управление приложениями-службами”, только теперь уже настраивается служба “Приложение-служба подключения к бизнес-данным”:
ex7
Как видим, этот вариант практически полностью (за исключением создания SecureStore) реализуем с помощью одного только SP Designer, однако имеет массу ограничений – нельзя таким образом работать с другими СУБД (например, Oracle), нельзя работать с объединением 2х и более таблиц – для этого нужно описывать модель работы с данными в коде, используя Visual Studio. В курсе 10175A SharePoint Application Development вторая лабораторная шестого урока посвящена именно такому созданию модели.

Вариант 2: доступ к внешней БД через powershell, работа с данными из БД как с объектами

Вот пример синхронизации списка SharePoint с данными из внешней БД Oracle.
Используется функция Get-DatabaseData, взято вот отсюда: http://technet.microsoft.com/ru-ru/magazine/hh855069.aspx
# Позволяет выполнять запросы к БД Oracle
function Get-DatabaseData {
[CmdletBinding()]
param (
  [string]$connectionString,
  [string]$query,
  [switch]$isSQLServer
)
if ($isSQLServer) {
  Write-Verbose 'in SQL Server mode'
  $connection = New-Object System.Data.SqlClient.SqlConnection
} else {
  Write-Verbose 'in OleDB mode'
  $connection = New-Object System.Data.OleDb.OleDbConnection
}
$connection.ConnectionString = $connectionString
#'Provider=OraOLEDB.Oracle;Data Source=(DESCRIPTION=(ADDRESS_LIST=(ADDRESS=(PROTOCOL=TCP)(HOST=host.host.ru)(PORT=1521)))(CONNECT_DATA=(SID=SOMESID)));User Id=USERID;Password=PASSWORD;'
#$connectionString = 'Provider=OraOLEDB.Oracle;Data Source=(DESCRIPTION=(ADDRESS_LIST=(ADDRESS=(PROTOCOL=TCP)(HOST=host.host.ru)(PORT=1521)))(CONNECT_DATA=(SID=SOMESID)));User Id=USERID;Password=PASSWORD;''
$command = $connection.CreateCommand()
$command.CommandText = $query
if ($isSQLServer) {
  $adapter = New-Object System.Data.SqlClient.SqlDataAdapter $command
} else {
  $adapter = New-Object System.Data.OleDb.OleDbDataAdapter $command
}
$dataset = New-Object System.Data.DataSet
$adapter.Fill($dataset)
return $dataset.Tables[0]
}
Источником данных может быть любая БД, разница только в строке подключения. Формат строки подключения для любой СУБД можно взять с http://connectionstrings.com/
Можно выполнять любые операции – INSERT, DELETE, CREATE и т.д. при наличии доступа к этой БД.
Ну и сам пример. Сравнивает данные о командах из таблицы БД с данными в списке “Alarm groups” по ID. Eсли команда переименована или создана, изменяет или создаёт новый элемент в списке. Если команда удалена – проверяет наличие связанных элементов из списка “Точки мониторинга” и при отсутствии связей удаляет элемент, иначе выдаёт оповещение:
$datarow = Get-DatabaseData -verbose -query "SELECT * FROM DUTY.group" -connectionString "Provider=OraOLEDB.Oracle;Data Source=(DESCRIPTION=(ADDRESS_LIST=(ADDRESS=(PROTOCOL=TCP)(HOST=host.host.ru)(PORT=1521)))(CONNECT_DATA=(SID=SOMESID)));User Id=USERID;Password=PASSWORD;"
$site = new-object Microsoft.SharePoint.SPSite("http://mysite")
$web = $site.OpenWeb("myweb")
$list = $web.Lists["Alarm groups"]
[Microsoft.SharePoint.SPListItemCollection]$list_items = [Microsoft.SharePoint.SPListItemCollection]$list.Items;
$message = "Запускаем процедуру синхронизации. Проверяем наличие новых или переименованных команд..."
log "Аварийные команды" $message
for( $i=1; $i -le $datarow[0];$i++) {
    $bool = 0
foreach( $list_item in $list_items ) {
    if ($list_item["ID0"] -eq $datarow[$i].Item(0)) {
  if ($datarow[$i].Item(1) -eq $list_item["Title"]) {
            $bool=1
   continue
   }
  else {
   $list_item["Title"] = $datarow[$i].Item(1)
            $list_item["_x0413__x041a__x0414_"] = "http://www.site.ru?groupId="+$datarow[$i].Item(0)+", "+$datarow[$i].Item(1)
   update $list_item
   $message = "Команда"+" "+$datarow[$i].Item(0)+" "+"переименована из"+" "+$list_item["Title"]+" в "+$datarow[$i].Item(1)
   log "Аварийные команды" $message
            $bool=1
            continue
   }
        }
}
    if (!($bool)) {
       $message = "Команда"+" "+$datarow[$i].Item(0)+" "+$datarow[$i].Item(1)+" "+"добавлена"
    log "Аварийные команды" $message
       $list_item = $list.Items.Add()
       $list_item["ID0"] = $datarow[$i].Item(0)
       $list_item["Title"] = $datarow[$i].Item(1)
       $list_item["_x0413__x041a__x0414_"] = "http://www.site.ru?groupId="+$datarow[$i].Item(0)+", "+$datarow[$i].Item(1)
       update $list_item
    }
}
$message = "Проверяем наличие удалённых команд"
log "Аварийные команды" $message
for($c = $list_items.Count - 1; $c -ge 0; $c--) {
    $bool = 0
    for( $i=1; $i -le $datarow[0];$i++) {
        if ( $list_items[$c]["ID0"] -eq $datarow[$i].Item(0)) {
            $bool=1
            continue
            }
        }
    if (!($bool)) {
    $title = $list_items[$c]["Title"]
    $coll_item_name = ""
$order_listx = $web.Lists["Точки мониторинга"]
    $queryx = new-object Microsoft.SharePoint.SPQuery
    $queryx.query = "<Where><Eq><FieldRef Name='_x0410__x0432__x0430__x0440__x04' /><Value Type='Lookup'>$title</Value></Eq></Where>"
$collectionx = $order_listx.GetItems( $queryx )
    if( $collectionx.count) {
        foreach( $coll_item in $collectionx){
            $title = $coll_item["Title"]
            $id = $coll_item["ID"]
            $coll_item_name = $coll_item_name+"<br>"+"<a href='http://mysite/myweb/Lists/PointsMonitoring/DispForm.aspx?ID=$id'>$title</a>"
            }
        $message = "ВНИМАНИЕ! Команда"+" "+$list_items[$c]["Title"]+" "+"была удалена из http://www.site.ru/DutyList, но не может быть удалена из списка Аварийных команд, т.к. остались ТМ со ссылкой на нее: "+$coll_item_name
        log "Аварийные команды" $message
        }
    else {
        $message = "Команда"+" "+$list_items[$c]["Title"]+" "+"была удалена из списка Аварийных команд"
        log "Аварийные команды" $message
        $list_items.Delete($c)
        }
    }
}
$list.Update()
$message = "Процедура синхронизации завершена"
log "Аварийные команды" $message

понедельник, 28 января 2013 г.

Создание отношений между списками


Довольно популярная задача, возникающая в рамках практически любого решения на SharePoint  – создание отношений между родительским и дочерним списками. К примеру “Проект – Задачи”, “Район – Улицы”, “Начальник – Подчиненные” и т.д.
Самый очевидный способ связать элементы разных списков – использовать поле “Подстановка” (lookup). На примере отношения “Проект – Задачи”:
Вариант 1: находясь на странице создания задачи (tasks/newform.aspx), выбираем обычным способом родительский проект из списка, это не требует никаких вмешательств в штатный функционал. Способ очень неудобный тем, что при большом количестве проектов приходится искать их в выпадающем списке.
Вариант 2: на странице просмотра проекта (projects/dispform.aspx) размещаем ссылку/кнопку, по клику на которую открывается форма создания задачи (tasks/newform.aspx) с уже заполненным полем “Проект”
Код ссылки/кнопки, который нужно вставить на странице projects/dispform.aspx:

<a href="/../Lists/Tasks/NewForm.aspx?ActID={$Param1}&amp;Source=https%3A%2F%2F..Lists%2Fprojects%2FDispForm%2Easpx&#063;ID={$Param1}">Создать задачу</a>
При этом должен быть заведен параметр в веб-части, в которой мы размещаем ссылку. Заведён так, как показано на рисунке.

Из переменной в адресной строке браузера этот параметр попадёт в tasks/newform.aspx , а атрибут “Source=…” нужен для корректного возврата после сохранения элемента в нужный projects/dispform.aspx
Для того, чтобы значение отобразилось в поле “подстановка” формы tasks/newform.aspx, необходимо вставить на эту страницу код на языке javascript – см. пример в конце статьи.

Отображение связанных записей в DispForm родительского списка

Для того, чтобы связанные записи были видны в форме, делаем следующее.
1. Создаём веб-часть представления дочернего списка в DispForm родительского списка.

 
Поигравшись с вкладкой “конструктор” на Ribbon-панели, можно придать веб-части удобоваримый вид. Например, если известно, что в рамках одного проекта обычно генерируется-назначается 5-6 задач, то очень хорошо подходит вот такой вид:

2. Необходимо добавить фильтр на представление, без него мы будем видеть все элементы списка Задачи. а нам нужны только связанные с конкретным проектом. ID проекта содержится в адресной строке браузера. Как создать параметр, получающий значение из адресной строки – было показано на первом рисунке.
Однако, если мы зададим условие фильтрации “Проект = Param1”, условие не сработает. Дело в том, что фактически мы сравниваем “1#;<название проекта>” и “1” и такое условие не может быть true. Для того, чтобы у нас было корректное поле для сравнения с параметром, нужно в свойствах списка, в параметрах поля-подстановки активировать дополнительное поле Проект:ИД

и уже это поле Проект:ИД использовать в условии фильтрации

Как скрыть поля связи (“lookup”, оно же “подстановка”) из форм

Если стоит задача скрыть поле из всех форм: создания, просмотра и редактирования – самый простой способ сделать это через SPD:

или непосредственно в браузере, последовательно – активировать управление типами содержимого в параметрах списка, зайти в тип содержимого, перевести столбец в “скрытый”. убрать управление типами содержимого.
Если нужно скрыть поле только из определённых форм, можно использовать утилиту SharePoint Manager
Она позволяет менять свойства SPField - ShowInEditForm, ShowInDisplayForm, ShowInNewForm. Лежит тут http://spm.codeplex.com/
Запускается на сервере с Sharepoint.

Можно отредактировать форму по умолчанию в дизайнере 2010 :
http://office.microsoft.com/ru-ru/sharepoint-designer-help/HA010378258.aspx
И, наконец, можно скрыть ненужные поля (или даже отобразить дополнительные) с помощью jquery, уже на стороне клиента, в браузере.


Пример установки поля “подстановка” в значение, полученное из адресной строки браузера и последующего скрытия строки с полем “подстановка”:

<script language="javascript" src="../frms/jquery-1.8.2.js" type="text/javascript"></script>
<script language="javascript" src="../frms/jquery.SPServices-0.7.2.js" type="text/javascript"></script>
<script type="text/javascript">
    // This javascript sets the default value of a lookup field identified
    // by <<FIELD DISPLAY NAME>> to the value stored in the querysting variable
    // identified by <<QUERYSTRING VARIABLE NAME>>
    // Customize this javascript by replacing <<FIELD DISPLAY NAME>> and
    // <<QUERYSTRING VARIABLE NAME>> with appropriate values.
    // Then just paste it into NewForm.aspx inside PlaceHolderMain
  
    _spBodyOnLoadFunctionNames.push("fillDefaultValues");
    function fillDefaultValues() {
      var qs = location.search.substring(1, location.search.length);
      var args = qs.split("&");
      var vals = new Object();
      for (var i=0; i < args.length; i++) {
        var nameVal = args[i].split("=");
        var temp = unescape(nameVal[1]).split('+');
        nameVal[1] = temp.join(' ');
        vals[nameVal[0]] = nameVal[1];
      }
      setLookupFromFieldName("Проект", vals["ID"]);
      $('#ctl00_m_g_e74ceaaa_2dcf_4cef_af99_40f05b6563c0_ff20_1_ctl00_ctl01').closest('tr').hide();   //   JQuery
    }
   
    function setLookupFromFieldName(fieldName, value) {
      if (value == undefined) return;
      var theSelect = getTagFromIdentifierAndTitle("select","Lookup",fieldName);
    // if theSelect is null, it means that the target list has more than
    // 20 items, and the Lookup is being rendered with an input element
      if (theSelect == null) {
        var theInput = getTagFromIdentifierAndTitle("input","",fieldName);
        ShowDropdown(theInput.id); //this function is provided by SharePoint
        var opt=document.getElementById(theInput.opt);
        setSelectedOption(opt, value);
        OptLoseFocus(opt); //this function is provided by SharePoint
      } else {
        setSelectedOption(theSelect, value);
      }
    }

    function setSelectedOption(select, value) {
      var opts = select.options;
      var l = opts.length;
      if (select == null) return;
      for (var i=0; i < l; i++) {
        if (opts[i].value == value) {
          select.selectedIndex = i;
          return true;
         }
       }
       return false;
     }
     
    function getTagFromIdentifierAndTitle(tagName, identifier, title) {
        var len = identifier.length;
        var tags = document.getElementsByTagName(tagName);
        for (var i=0; i < tags.length; i++) {
          var tempString = tags[i].id;
          if (tags[i].title == title && (identifier == "" || tempString.indexOf(identifier) == tempString.length - len)) {
            return tags[i];
          }
        }
        return null;
    }
    </script>

Sharepoint–разное накопившееся


  • Отсутствует в меню Действия узла -Изменить страницу : ?toolpaneview=2
  • Активности WorkFlow:
  1. Если возникла проблема, что в дизайнере активности есть, но не добавляются...

    Решение проблемы:
    --- ЦИТАТА----
    I found the problem. Step 2 of the install guide states to update the web.config file (usually c:\inetpub\wwwroot\wss\VirtualDirectories\80). I modified this file, but it was not the correct web.config file. I found a blog (http://blogs.msdn.com/sharepointdesigner/archive/2007/03/15/adding-activities-to-spd.aspx) that gives more detailed instructions on how to update the correct web.config file. I recommend updating the install guide with the following instructions for updating the web.config file.
    1. Open the IIS console.
    2. Find the web site that is your SharePoint web under the "Web sites" node and select it.
    3. Right click and choose "Open".
    4. In the folder window, open the web.config file.
    5. In web.config, browse to the bottom section and look for <System.Workflow.ComponentModel.WorkflowCompiler>. Inside that tag there should be an <authorizedTypes> section and an entry for each set of types. Add the following line of XML:
    <authorizedType Assembly="DP.Sharepoint.Workflow, Version=1.0.0.0, Culture=neutral, PublicKeyToken=0298457208daed83" Namespace="DP.Sharepoint.Workflow" TypeName="*" Authorized="True" />
    --- ЦИТАТА----
возможен другой токен сборки, для проверки выполняем: <путь>\gacutil /l DP.Sharepoint.Workflow
  1. Кроме того, бывает что активности даже не появляются в дизайнере.
В этом случае необходимо сделать следующее: If you have a non-english installation of sharepoint, you need to copy <file>.ACTIONS from [SP HiVE]\TEMPLATE\1033\Workflow to [SP HiVE]\TEMPLATE\1049\Workflow
  1. Кроме того, иногда РП просто отказывается открываться:
Помогал запуск Дизайнера из-под системной учетки
  • Включить подробный вывод ошибок:
в web.config нужного сайта изменить <customErrors mode="On" /> на <customErrors mode="Off" />
<SafeMode MaxControls="200" CallStack="false"... на <SafeMode MaxControls="200" CallStack="true"...

для подробного вывода ошибок рабочего процесса: http://blogs.technet.com/b/victorbutuza/archive/2009/02/26/how-to-enable-workflow-tracing-debug.aspx
вкратце:
в web.config нужного сайта включить:
после </System.Workflow.ComponentModel.WorkflowCompiler>
   <system.diagnostics>
      <switches>
          <add name="System.Workflow LogToFile" value="1" />
          <add name="System.Workflow.Runtime" value="All" />
          <add name="System.Workflow.Runtime.Hosting" value="All" />
          <add name="System.Workflow.Runtime.Tracking" value="All" />
          <add name="System.Workflow.Activities" value="All" />
          <add name="System.Workflow.Activities.Rules" value="All" />     
      </switches>
  </system.diagnostics>

и искать результат в c:\windows\system32\inetsrv, workflowtrace.log

Повышение общего уровня логирования
stsadm.exe -o setlogginglevel -category "General" -tracelevel Verbose
stsadm.exe -o setlogginglevel -category "Workflow Infrastructure" -tracelevel Verbose
stsadm.exe -o setlogginglevel -category "Workflow Features" -tracelevel Verbose
Понижение уровня (чтобы не загадить логами весь диск)
stsadm.exe -o setlogginglevel -category "General" -tracelevel Monitorable
stsadm.exe -o setlogginglevel -category "Workflow Infrastructure" -tracelevel Monitorable
stsadm.exe -o setlogginglevel -category "Workflow Features" -tracelevel Monitorable
Чтобы изменения вступили в силу, нужно перезапустить службу "Windows SharePoint Services Tracing service"

Посмотреть куда пишутся логи
You can check your current log folder path at the
Central Administration\Operations\(below the Logging and Reporting section)Diagnostic logging\in the Trace log section in the Path editbox.)
  • В 2010 нет ссылки на "настройку групп для узла". Но её можно открыть по ссылке "узел/_layouts/permsetup.aspx"
  • Удалить лишнюю информацию из lookup полей перенесённых в Excel:
    нужно создать макрос с именем Auto_Open, это позволит ему выполняться при открытии документа.
    Код макроса следующий:
        Cells.Replace What:="*;#", Replacement:="", LookAt:=xlPart, SearchOrder _
            :=xlByRows, MatchCase:=False, SearchFormat:=False, ReplaceFormat:=False
           Application.OnTime Now + TimeValue("00:00:06"), "Auto_Open"
  • Как раскрасить календарь: http://www.planetwilson.co.uk/sharepoint-2013-colour-calendar/ 
  • Как раскрасить список:
    <script> 
    var colors = new Array(
                      new Array( "принято с замечаниями", "#FF0000", "#000000" ),
                      new Array( "принято"              , "#00FF00", "#0000FF" ),
                      new Array( "не принято"           , "#0000FF", "#000000" )
                                  );
    function getObjInnerText (obj) 

          return (obj.innerText) ? obj.innerText : (obj.textContent) ? obj.textContent : ""; 
    }  
    function runOnLoad() {
          nodes = document.getElementsByTagName("td"); 
            for( var i = 0; i < nodes.length; i++ ) 
            { 
                var itext = getObjInnerText( nodes[i] )
                for( i1 in colors ) {
                      if( colors[i1][0] == itext ) {
                            nodes[i].style.background = colors[i1][1]
                            nodes[i].style.color      = colors[i1][2]
                      }
                }
             }
    }
  • function ExpGroupRenderData( htmlToRender, groupName, isLoaded ) {      var tbody = document.getElementById( "tbod" + groupName + "_" )      var wrapDiv = document.createElement( "DIV" )
          wrapDiv.innerHTML = "<TABLE><TBODY id=\"tbod" + groupName + "_\" isLoaded=\"" + isLoaded + "\">" + htmlToRender + "</TBODY></TABLE>";     
          tbody.parentNode.replaceChild( wrapDiv.firstChild.firstChild, tbody );
          runOnLoad();
    }
    //_spBodyOnLoadFunctionNames.push("runOnLoad");
    </script>
  • РП в дизайнере при назначении задачи пользователю не дает подставить в качестве адресата поле с типом "Пользователь", если там разрешен множественный выбор. Но достаточно убрать его, потом поставить его в РП, а потом вернуть как было и множественный выбор работает ) Задачи уходят нескольким людям...Активность называется "Назначить форму группе", например. В этом случае РП ожидает закрытия всех задач обычно, он это если указывать адресатов по одному, а не подсовывая в качестве списка адресатов поле со множственным выбором. Хтя скорее всего корректно отработает. Если имеем дело с активностью "получить данные от пользователя", то РП может пойти дальше и после закрытия первой задачи.
    •Если рабочие процессы ведут себя странно, а именно: при создании/изменения элемента висят в состоянии "Запуск", отрабатывая нормально только в случае запуска вручную - предположительно помогает очистка корзины семейства веб-узлов. Кроме очистки корзины было сделано:
  • iisreset
    Attempting stop...
    Internet services successfully stopped
    Attempting start...
    Internet services successfully restarted
    stsadm -o getproperty -pn job-workflow -url http://…
    <Property Exist="No" />
    stsadm -o setproperty -pn job-workflow -pv "Every 1 minutes between 0 and 59" -url http://…
    Операция успешно завершена.
    stsadm -o getproperty -pn workflow-eventdelivery-throttle
    <Property Exist="Yes" Value="15" />
    stsadm -o setproperty -pn workflow-eventdelivery-throttle -pv "20"
    Операция успешно завершена.
    •Вот тут: /_layouts/wrkmng.aspx можно поглядеть выполняющиеся РП.
    •Рабочие процессы на элементы, добавляющиеся по email:
    Вкратце, проблема связана с тем что эти записи добавляются из-под Системной Учетной Записи - http://support.microsoft.com/kb/953289/ru
    фикс лечит проблему
    •Привязка РП к опросу не прокатит. Тут об этом написано Привязка РП к типу "Опрос"
  • В 2010ом всякий рабочий процесс ведёт свой журнал в "Журнале рабочего процесса". Там всегда можно посмотреть хронологию выполнения экземпляров РП.
  • На список Task (Задачи) можно повесить РП, но нельзя его просмотреть/запустить руками из менюшек. Для просмотра можно создать представление, в которое включить поля - рабочие процессы. Для запуска руками - воспользоваться ссылкой /_layouts/IniWrkflIP.aspx?List={ae2420f3-b6e8-49b6-a458-c071847121b7}&ID=814&TemplateID={25b34c92-7c30-433d-8e76-d215c8142a89} айдишники списка достать из того же "Журнал рабочего процесса", в нем TemplateID это "Идентификатор сопоставления рабочего процесса"
  • По написанию рабочих процессов. http://office.microsoft.com/ru-ru/sharepoint-designer-help/CH010373544.aspx

воскресенье, 27 января 2013 г.

Завершение экземпляров WorkFlow

Скрипт завершает все выполняющиеся экземпляры РП и при необходимости выполняет перезапуск.
Иногда он выполняет рестарт даже если не указывать -restart, во избежание этого нужно закомментарить выделенный красным фрагмент.
 
Может также удалять старые выполняющиеся экземпляры РП. Отслеживать выполнение можно тут: _layouts/wrksetng.aspx?List=%7B8E4A6033%2DFD1B%2D4BBA%2DAE60%2D6D2869D3A28B%7D (ID списка ваш)
 
Пример использования:
Видим картинку, хотим удалить старые экземпляры процесса "Оповещения (предыдущая версия: 23.09.2011 14:01:26)"
 
wf

 
Запускаем ./Cancel-SPWorkflow -site "https://skr.mts.ru/it/d/dpp/its" -List "Задачи" -Workflow "Оповещения (предыдущая версия: 23.09.2011 14:01:26)"
 
Дожидаемся окончания выполнения скрипта и видим:
 
wf2

 
 
##################################################################################
#
#
#  Script name: Cancel-SPWorkflow.ps1
#
#  Author:       Mattias.Karlsson@zipper.se
#  Homepage:     www.mysharepointofview.com
#
#
##################################################################################

param([string]$site, [string]$List, [string]$Workflowname, [switch]$Restart, [switch]$IncludeRunning, [switch]$help)
[void][System.Reflection.Assembly]::LoadWithPartialName("Microsoft.SharePoint")
function GetHelp() {
$HelpText = @"
DESCRIPTION:
NAME: Cancel-SPWorkflow
Cancel SharePoint Workflows and restart them if wanted
PARAMETERS:
-Site    Url to the SharePoint Site
-List    Name of the list
-Workflowname  Name of the workflow
-Restart  Optional if you want to restart the workflows
-IncludeRunning  Optional if you want to include running workflows
-help     Displays the help topic
SYNTAX:
Cancel-SPWorkflow -help
Displays the help topic for the script
Cancel-SPWorkflow -site http://yourSPSite -list Listname -Workflowname YourWorkflow
Terminates all workflows with the specific workflow name
Cancel-SPWorkflow -site http://yourSPSite -list Listname -Workflowname YourWorkflow -Includerunning
Terminates all workflows with the specific workflow name and include running workflows
Cancel-SPWorkflow -site http://yourSPSite -list Listname -Workflowname YourWorkflow -restart
Terminates all workflows with the specific workflow name and then restart the workflow for all items
"@
$HelpText
}
function Get-SPSite([string]$site) {
New-Object Microsoft.SharePoint.SPSite($site)
}
function Get-SPWeb([string]$site) {
$SPSite = Get-SPSite $site
$SPSite.OpenWeb()
}
function Get-SPList([string]$site, [string]$List){
$OpenWeb = Get-SPWeb $site
$OpenWeb.Lists[$List]
}
function CancelWF($site, $List, $Workflowname, [switch]$Restart, [switch]$IncludeRunning) {
$SPList = Get-SPList $site $List
    Write-host $SPList.WorkflowAssociations
        if(($SPList.WorkflowAssociations | where {$_.Name -eq $Workflowname}) -gt $null){
        TerminateWF $site $List $Workflowname $Restart $IncludeRunning
    }
    else{
            Write-host "The Specifed Workflow does not exist in the list"
    }
}       
function StartWF($Site, $List, $Workflowname){
    $SPList = Get-SPList $site $List
    $WFAssociation = $SPList.WorkflowAssociations | where {$_.name -eq $Workflowname}
    foreach($item in $SPList.items) {
        $StartWorkflow = $SPList.Parentweb.site.workflowmanager.startworkflow($item, $WFAssociation, $WFAssociation.AssociationData)
    }
}
function TerminateWF($Site, $List, $Workflowname, $Restart, $IncludeRunning) {
    $SPList = Get-SPList $site $List
    foreach($item in $SPList.items) {
  if($item.Workflows.count -gt 0){
   foreach($Workflow in $item.workflows){
                if(($SPList.WorkflowAssociations | where {$_.id -eq $Workflow.AssociationId}).name -eq $Workflowname){
                    if($IncludeRunning){
         [Microsoft.SharePoint.Workflow.SPWorkflowManager]::CancelWorkflow($Workflow)
        }
                    else{
                        if(($Workflow.Internalstate -ne "Running") -and ($Workflow.Internalstate -ne "Completed") -and ($Workflow.Internalstate -ne "Cancelled")) {
[Microsoft.SharePoint.Workflow.SPWorkflowManager]::CancelWorkflow($Workflow)
                            } 
                      }
      }
   }
  }
}
    if($Restart){
        StartWF $Site $List $Workflowname
   }
   Write-host "Operation completed successfully" $TerminatedWFs
}
if($help) { GetHelp; Continue }
if($Site -AND $List -AND $Workflowname) { CancelWF -Site $Site -List $List -Workflowname $Workflowname -Restart $Restart -IncludeRunning $IncludeRunning}
else{ GetHelp; Continue}

“Отдел Х” захотел себе “сайт”. Создать для них веб приложение или семейство узлов?

Однозначного ответа нет, но вот такая табличка возможно поможет склониться к правильному варианту:

 

Веб-приложение

Семейство сайтов

Сайт внутри семейства

Возможность использовать для хранения контента отдельную БД

+

+

-

Возможность иметь различные СУС-КПЭ (доступность, восстановление, частота резервного копирования) в зависимости от контента

+

+

-

Возможность использовать отдельный пул приложений IIS

+

-

-

Возможность использовать уникальное доменное имя

+

-

-

Наличие собственных настроек/политик (группы доступа, корзина, др.)

+

+

-

Возможность настройки уникальных параметров веб-приложения (например тип аутентификации)

+

-

-

Возможность создания логического разделения контента сайтов

-

+

+

Возможность совместного использования шаблонов и навигации

-

+

+

Независимое развертывание решений

+

-

-

 

 

 

 

Так же стоит отметить, что в случае "разделения" узлов на приложения или семейства улучшается управляемость вследствие сокращения количества групп SharePoint и увеличения количества администраторов (у приложения или семейства узлов один администратор, соответственно при увеличении числа семейств в каждом - свой).

Sharepoint – частный случай решения “проблемы больших списков”

Пользователи Sharepoint очень любят наполнять списки данными. Часто это приводит вот к такой картине:

pic1

Пороговое значение списка 20 тыс. установлено в Администраторской консоли, обычно оно ещё меньше – всего 5 тыс. элементов.

После того как значение элементов списка превысило допустимый порог, в списке отключаются возможности фильтрации по полям, становится очень сложно им пользоваться, найти что-либо в таком списке можно только встроенным в SharePoint поиском.

В целом такое ограничение вполне оправдано. Список – это не таблица в БД, из него не выбрать данные select’ом, и необходимо осознанно подходить к наполнению списка данными, настройке представлений на списке. При бесконтрольном добавлении данных список становится неуправляемым, пользователям становится сложно искать в нём необходимые данные.

Однако бывают ситуации, когда большой список просто необходим. Ниже я опишу, что можно сделать для комфортной работы с таким списком.

Первое: необходимо отключить для отдельно взятого списка ограничение по порогу. Для всех остальных списков это ограничение продолжит работать, и администратору не придётся волноваться, что вышедшие из-под контроля пользователя за неделю заполнят всё имеющееся дисковое пространство на SQL-сервере Sharepoint.
Отключить ограничение можно установкой свойства isThrottled в “false”. Для этого нужно под учёткой Администратора фермы зайти на сервер, запустить командную консоль SharePoint и выполнить:
$list.EnableThrottling = $false
$list.Update()

После этого для списка будут доступны все возможности фильтрации, как если бы превышения порога не было. Впрочем, предупреждение о превышении порога в меню “Параметры списка” останется.

Второе: нужно настроить представления для комфортной работы пользователей. Если нам известно, что в основном определённые группы пользователей работают с определёнными группами записей, нужно создать для них фильтрованные представления. Например, если в нашем списке есть поле
“Регион”, можно создать представления под названием “Все записи – Магадан”, “Все записи – Колыма” и т.д., отфильтрованные по полю “Регион”.

Переключение между представлениями для начинающего пользователя Sharepoint не так очевидно, поэтому хорошо бы вывести ссылки на эти представления на отдельную страницу, или на меню быстрого запуска, или отдельной веб-частью в представление “по умолчанию”. Ни рисунке показан пример того, как ссылки на представления перечислены в веб-части “Редактор контента”

pic2

Третье и самое главное: штатный поиск Sharepoint не умеет искать элементы по части слова или маске. Так, например, если у нас есть элементы с названиями “первый”, “второй”, “третий”, то поиск “перв” или “перв*” ничего не даст. Однако, как показывает практика, это самый необходимый тип поиска с точки зрения пользователя, “затерявшегося” в большом списке. Такой поиск по одному полю достаточно просто реализуется с использованием пары веб-частей и Sharepoint Designer.

Создадим представление. Например, на основе существующего “Все записи – Магадан”. Назовём его “Магадан – фильтр записей”, и разместим в верхней части страницы веб-часть под названием “Фильтр текста”

Далее выделяем веб-часть с данными списка и добавляем параметр для веб-части с произвольным именем:

pic0

Затем кликаем на “Добавить подключение”, и соединяем веб-части, как показано на рисунках:

pic3

pic4pic5

И добавляем фильтр

pic6

После этого любой текст, введённый в фильтр, после нажатия “Ввод” ограничит количество записей.