Skip to content
LibIms_Steps.ttcn 64.8 KiB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000
/**
 * @author STF471
 * @version $Id$
 * @desc This module provides the steps used by the test component for SIP-IMS tests.
 *       This module is part of LibImsV3.
 */
module LibIms_Steps {
    // LibSip
    import from LibSip_SIPTypesAndValues all;
    import from LibSip_Templates all;
    import from LibSip_Steps all;
    import from LibSip_PIXITS all;
    import from LibSip_Interface all;
    import from LibSip_MessageBodyTypes all;
    import from LibSip_Common all;
    // LibIms
    import from LibIms_Templates all;
    import from LibIms_Interface all;
    import from LibIms_SIPTypesAndValues all;

    group externalfunctions {
    } // end group externalfunctions
    group parameterOperations {

        /**
         * @desc Adds new Via parameter
         * @param p_message (request) SIP message to be used to prepair via header
         */
        function f_addNewViaParameter(
            in Request p_message
        ) runs on ImsComponent  {
            var integer v_intVia;
            var integer i := 0;
            if (isvalue(p_message.msgHeader.via)) {

                v_intVia := lengthof(p_message.msgHeader.via.viaBody);
                while (i < v_intVia) {
                    p_message.msgHeader.via.viaBody[v_intVia] := p_message.msgHeader.via.viaBody[v_intVia - 1];
                    v_intVia := v_intVia - 1;
                }
                vc_branch := c_branchCookie & f_getRndTag();
                p_message.msgHeader.via.viaBody[0] := valueof(m_ViaBody_currIpaddr(vc_branch, vc_userprofile));
                vc_via := p_message.msgHeader.via;
            }
        }

        /**
         * @desc Removes own Via parameter
         * @param p_message (request) SIP message to be used to prepair via header
         */
        function f_removeOwnViaParameter(
            in Response p_message
        ) runs on ImsComponent  {
            var integer v_intVia;
            var Via v_via := c_empty_Via;
            var integer i := 0;

            if (ispresent(p_message.msgHeader.via)) {

                v_intVia := lengthof(p_message.msgHeader.via.viaBody) - 1;

                for (i := 0; i < v_intVia; i := i + 1) {
                    v_via.viaBody[i] := p_message.msgHeader.via.viaBody[i + 1];
                }
                vc_via := v_via;
            }
        }

        /**
         * @desc Adds new RecordRoute parameter and removes Route parameter
         * @param p_message (request) SIP message to be used to prepair via header
         */
        function f_addNewRecordRouteAndRemoveRoutParameter(
            in Request p_message
        ) runs on ImsComponent  {
            var integer v_intRoute, v_intRecordRoute;
            var integer i := 1;
            var RouteBody v_route1;
            var Route v_newRoute;
            v_newRoute.fieldName := ROUTE_E;
            if (isvalue(p_message.msgHeader.route)) {
                v_intRoute := lengthof(p_message.msgHeader.route.routeBody);
                v_route1 := p_message.msgHeader.route.routeBody[0];
                while (i < v_intRoute) {
                    v_newRoute.routeBody[i - 1] := p_message.msgHeader.route.routeBody[i];
                    i := i + 1;
                }
                vc_route := v_newRoute;
            }

            if (isvalue(p_message.msgHeader.recordRoute)) {

                v_intRecordRoute := lengthof(p_message.msgHeader.recordRoute.routeBody);
                while (0 < v_intRecordRoute) {
                    p_message.msgHeader.recordRoute.routeBody[v_intRecordRoute] := p_message.msgHeader.recordRoute.routeBody[v_intRecordRoute - 1];
                    v_intRecordRoute := v_intRecordRoute - 1;
                }
                p_message.msgHeader.recordRoute.routeBody[0] := v_route1;
                vc_recordRoute := valueof(p_message.msgHeader.recordRoute);
            }
        }

        /**
         * @desc Adds new RecordRoute parameter (IMS)
         * @param p_rr The recordRoute parameter to add
         */
        function f_addNewRecordRouteIMS(
            in RecordRoute p_rr
        ) runs on ImsComponent 
        return template RecordRoute  {
            var integer v_intRecordRoute;
            var template RecordRoute v_recordRoute := omit;

            v_intRecordRoute := lengthof(p_rr.routeBody);
            if (v_intRecordRoute > 0) {
                while (0 < v_intRecordRoute) {
                    p_rr.routeBody[v_intRecordRoute] := p_rr.routeBody[v_intRecordRoute - 1];
                    v_intRecordRoute := v_intRecordRoute - 1;
                }
                p_rr.routeBody[0] := valueof(m_routeBody_currIpAddr(vc_userprofile));
                vc_recordRoute := p_rr;
                return vc_recordRoute;
            }
            else {
                return (v_recordRoute);
            }
        }

    } // end group parameterOperations

    group fieldOperations {

        /**
         * @desc Sets BYE header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersBYE(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user
        ) runs on ImsComponent  {
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            vc_route := f_route(); // update the route header field depending on vc_boo_route
            vc_recordRoute := f_recordroute(); // update the route header field depending on vc_boo_route
            LibSip_Steps.f_setHeadersBYE(p_cSeq_s);
        } // end f_setHeadersBYE

        /**
         * @desc Sets CANCEL header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         */
        function f_setHeadersCANCEL(
            inout CSeq p_cSeq_s
        ) runs on ImsComponent  {
            vc_route := f_route(); // update the route header field depending on vc_boo_route
            vc_recordRoute := f_recordroute(); // update the route header field depending on vc_boo_route
            LibSip_Steps.f_setHeadersCANCEL(p_cSeq_s);
        } // end f_setHeadersCANCEL

        /**
         * @desc Sets headers for forward request from AS in case if AS acts as Proxy
         * @param p_cSeq_s The current cSeq
         * @param p_proxyMode true = proxyMode, false = B2BMode
         */
        function f_setHeadersForwardRequestFromAS(
            inout CSeq p_cSeq_s,
            boolean p_proxyMode
        ) runs on ImsComponent  {
            var Request v_request;
            v_request := vc_request;

            if (p_proxyMode) {
                vc_requestUri2 := v_request.requestLine.requestUri;
                vc_to := v_request.msgHeader.toField;
                vc_contact := v_request.msgHeader.contact;
                f_addNewViaParameter(v_request);
                f_addNewRecordRouteAndRemoveRoutParameter(v_request);
            }
            else {
                // B2Bmode
                vc_contact := m_Contact(m_SipUrl_contactIpaddr(vc_userprofile));
                if (PX_SEED) {
                  vc_callId := {
                      fieldName := CALL_ID_E,
                      callid := f_getRndCallId()
                  };
                } else {
                  vc_callId := {
                      fieldName := CALL_ID_E,
                      callid := f_getRndCallId() & c_AT & vc_userprofile.currIpaddr
                  };
                }
            }
        } // end function f_setHeadersForwardRequest

        /**
         * @desc Sets header fields for forward request from AS in case if AS acts as Proxy
         */
        function f_setHeadersForwardResponseFromAS(
        ) runs on ImsComponent  {
            var Response v_response;
            v_response := vc_response;

            vc_caller_To := v_response.msgHeader.toField;
            vc_to := v_response.msgHeader.toField;
            vc_contact := v_response.msgHeader.contact;

            if (ispresent(v_response.msgHeader.recordRoute)) {
                vc_recordRoute := v_response.msgHeader.recordRoute;
            }
            f_removeOwnViaParameter(v_response);
        } // end function f_setHeadersForwardResponse

        /**
         * @desc Sets Invite header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersINVITE(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            vc_to := f_initToHeader(p_to_user); // init of vc_to using userProfile identifier
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            if (vc_boo_route) {
                vc_route := m_route_interface(vc_interfaceprofile);
            }
            if (vc_boo_recordRoute) {
                vc_recordRoute := m_recordRoute_currIpAddr(vc_userprofile);
            }

            LibSip_Steps.f_setHeadersINVITE(p_cSeq_s);
        } // end f_setHeadersINVITE

        /**
         * @desc Sets Invite header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersUPDATE(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            vc_to := f_initToHeader(p_to_user); // init of vc_to using userProfile identifier
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            if (vc_boo_route) {
                vc_route := m_route_interface(vc_interfaceprofile);
            }
            if (vc_boo_recordRoute) {
                vc_recordRoute := m_recordRoute_currIpAddr(vc_userprofile);
            }

            LibSip_Steps.f_setHeadersUPDATE(p_cSeq_s);
        } // end f_setHeadersUPDATE

        /**
         * @desc Sets Message header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersMESSAGE(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            vc_to := f_initToHeader(p_to_user); // init of vc_to using userProfile identifier
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            LibSip_Steps.f_setHeadersMESSAGE(p_cSeq_s);
        } // end f_setHeadersMESSAGE

        /**
         * @desc Sets Notify header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersNOTIFY(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            LibSip_Steps.f_setHeadersNOTIFY(p_cSeq_s);

            vc_branch := c_branchCookie & f_getRndTag();
            vc_via := {
                fieldName := VIA_E,
                viaBody := {valueof(m_ViaBody_virtual_XCSCF(vc_branch, vc_userprofile))}
            };

            vc_to := f_initToHeader(p_to_user); // init of vc_to using userProfile identifier
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            // contact header initialization for sending of NOTIFY from CSCF component
            vc_contact := m_Contact(m_SipUrl_currIpaddr_CSCF(vc_userprofile));
        } // end f_setHeadersNOTIFY

        /**
         * @desc Sets Publish header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersPUBLISH(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            LibSip_Steps.f_setHeadersPUBLISH(p_cSeq_s);

            vc_branch := c_branchCookie & f_getRndTag();
            vc_via := {
                fieldName := VIA_E,
                viaBody := {valueof(m_ViaBody_virtual_XCSCF(vc_branch, vc_userprofile))}
            };

            vc_to := f_initToHeader(p_to_user); // init of vc_to using userProfile identifier
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            if (vc_boo_route) {
                vc_route := m_route_interface(vc_interfaceprofile);
            }
            if (vc_boo_recordRoute) {
                vc_recordRoute := m_recordRoute_currIpAddr(vc_userprofile);
            }

            // contact header initialization for sending of PUBLISH from CSCF component
            vc_contact := m_Contact(m_SipUrl_currIpaddr_CSCF(vc_userprofile));
        } // end f_setHeadersPUBLISH

        /**
         * @desc Sets Subscribe header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersSUBSCRIBE(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in SipUrl p_from_user
        ) runs on ImsComponent  {
            vc_to := {
                fieldName := TO_E,
                addressField := {
                    nameAddr := {
                        displayName := omit,  // optional charstring
                        addrSpec := p_to_user // SipUrl
                    }
                },  // end addressField
                toParams := omit
            };
            vc_from := f_initFromHeader(p_from_user, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            LibSip_Steps.f_setHeadersSUBSCRIBE(p_cSeq_s);
        } // end f_setHeadersSUBSCRIBE

        /**
         * @desc Sets headers for ReINVITE method
         * @param p_cSeq_s The current cSeq
         * @param p_orginatingSide true in case of Re-INVITE is send from the orgination endpoint otherwise false.
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersReINVITE(
            inout CSeq p_cSeq_s,
            in boolean p_orginatingSide,
            in SipUrl p_to_user
        ) runs on ImsComponent  {
            var integer v_tmp, i, j, v_nbroute;
            var Request v_request;
            v_request := vc_request;

            f_setHeadersGeneral(p_cSeq_s, "INVITE"); // cseq, contact, branch, via
            vc_reqHostPort := vc_requestUri.components.sip.hostPort;

            vc_requestUri := p_to_user;

            if (p_orginatingSide) {
                vc_to := vc_caller_To;
                vc_from := vc_caller_From;
            }
            else {
                vc_to := vc_callee_To;
                vc_from := vc_callee_From;

                // get route from previous ACK request
                // Route Management
                if (isvalue(v_request.msgHeader.recordRoute)) {
                    vc_recordRoute := valueof(v_request.msgHeader.recordRoute);
                    v_nbroute := lengthof(vc_recordRoute.routeBody);
                    // copy and reverse the order of the routes in route header
                    for (i := 0; i <= (v_nbroute - 1); i := i + 1) {
                        j := v_nbroute - 1 - i;
                        vc_route.routeBody[j] := vc_recordRoute.routeBody[i];
                    }
                    vc_route.fieldName := ROUTE_E;
                    vc_boo_recordRoute := true;
                    vc_boo_route := true;
                }
                else {
                    vc_boo_recordRoute := false;
                    vc_boo_route := false;
                }
            }

            v_tmp := str2int(vc_sdp_local.origin.session_id);
            vc_sdp_local.origin.session_id := int2str(v_tmp + 1);
            v_tmp := str2int(vc_sdp_local.origin.session_version);
            vc_sdp_local.origin.session_version := int2str(v_tmp + 1);
        } // end function f_setHeadersReINVITE

        /**
         * @desc Sets component variables related to message header fields when sending requests from the home I-CSCF (TS) to the visited P-CSCF (SUT) (message type independent: CSeq, contact, via), function uses information from userprofile and interfaceprofile
         * @param p_cSeq_s The CSeq parameter
         * @param p_method The method name for cSeq header field
         */
        function f_setHeadersGeneral_ICSCF(
            inout CSeq p_cSeq_s,
            in charstring p_method
        ) runs on SipComponent  {
            p_cSeq_s.fieldName := CSEQ_E;
            p_cSeq_s.seqNumber := p_cSeq_s.seqNumber + 1;
            p_cSeq_s.method := p_method;
            vc_cSeq := p_cSeq_s;

            vc_contact := m_Contact(m_SipUrl_contactIpaddr(vc_userprofile));

            vc_branch := c_branchCookie & f_getRndTag();
            vc_branch_ICSCF := c_branchCookie & f_getRndTag();

            vc_via := {
                fieldName := VIA_E,
                viaBody := {
                    valueof(m_ViaBody_currIpaddr(vc_branch, vc_userprofile)), 
                    valueof(m_ViaBody_virtualUEinPCSCF(vc_branch_ICSCF, vc_userprofile))
                }
            };
        } // end function f_setHeadersGeneral_ICSCF

        /**
         * @desc Sets header field for the next outgoing REGISTER message from the visited P-CSCF to the home I-CSCF
         * @param p_cSeq_s The CSeq parameter to be applied
         */
        function f_setHeaders_REGISTER_PCSCF(
            inout CSeq p_cSeq_s
        ) runs on SipComponent  {
            var SemicolonParam_List v_params;

            f_setHeadersGeneral(p_cSeq_s, "REGISTER"); // cseq, contact, branch, via
            vc_requestUri := {
                scheme := c_sipScheme,
                components := {
                    sip := {
                        userInfo := omit,
                        hostPort := {
                            host := vc_userprofile.registrarDomain,
                            portField := omit
                        }
                    }
                },
                urlParameters := omit,
                headers := omit
            };

            vc_reqHostPort := vc_requestUri.components.sip.hostPort;

            if (PX_SEED) {
              vc_callId := {
                  fieldName := CALL_ID_E,
                  callid := f_getRndCallId()
              };
            } else {
              vc_callId := {
                  fieldName := CALL_ID_E,
                  callid := f_getRndCallId() & c_AT & vc_userprofile.currIpaddr
              };
            }
            vc_callIdReg := vc_callId;

            vc_to := valueof(m_To(m_SipUrl_currDomain(vc_userprofile)));
            v_params := {
                {
                    id := c_tagId,
                    paramValue := {
                        tokenOrHost := f_getRndTag()
                    }
                }
            };
            vc_from := {
                fieldName := FROM_E,
                addressField := vc_to.addressField,
                fromParams := v_params
            };

            if (not vc_firstREGISTER_sent) {
                v_params := {
                    {
                        id := c_expiresId,
                        paramValue := {
                            tokenOrHost := c_shortRegistration
                        }
                    }
                };
                vc_contact.contactBody.contactAddresses[0].contactParams := v_params;
            }

            vc_firstREGISTER_sent := true; // f_setHeaders_Register is called in deREGISTER function
            vc_authorization := {
                fieldName := AUTHORIZATION_E,
                body := {f_calculatecCredentials_empty(vc_userprofile)}
            };
        } // end function setHeaders_REGISTER_PCSCF

        /**
         * @desc Sets REFER header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_to_user The selected user's SIP URL
         */
        function f_setHeadersREFER(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user
        ) runs on ImsComponent  {
            vc_requestUri := p_to_user; // Request URI of Invite is identical with To header
            if (vc_boo_route) {
                vc_route := m_route_interface(vc_interfaceprofile);
            }
            if (vc_boo_recordRoute) {
                vc_recordRoute := m_recordRoute_currIpAddr(vc_userprofile);
            }

            LibSip_Steps.f_setHeadersREFER(p_cSeq_s);
        } // end f_setHeadersREFER

        /**
         * @desc Sets REFER header fields (IMS addresses) extension of general settings from LibSip basic function
         * @param p_cSeq_s The current cSeq
         * @param p_uri The SipUrl for request URI and To header
         */
        function f_setHeadersREFER_conf(
            inout CSeq p_cSeq_s,
            in SipUrl p_uri
        ) runs on ImsComponent  {
            vc_to := {
                fieldName := TO_E,
                addressField := {
                    nameAddr := {
                        displayName := omit,  // optional charstring
                        addrSpec := p_uri // SipUrl
                    }
                },  // end addressField
                toParams := omit
            };

            vc_from := f_initFromHeader(p_uri, f_getRndTag()); // init of vc_from using userProfile identifier
            vc_requestUri := p_uri; // Request URI of Invite is identical with To header
            if (vc_boo_route) {
                vc_route := m_route_interface(vc_interfaceprofile);
            }
            if (vc_boo_recordRoute) {
                vc_recordRoute := m_recordRoute_currIpAddr(vc_userprofile);
            }

            LibSip_Steps.f_setHeadersREFER(p_cSeq_s);
        } // end f_setHeadersREFER_conf

    } // end group fieldOperations

    group awaitingMessage {
    }
    
    group sendMessage {

        /**
         * @desc Sends PRACK message
         * @param p_mb The message body of the PRACK message
         * @param p_rack_cseq The cSeq of the PRACK message
         */
        function f_sendPRACK_sdp(
            template(value) MessageBody p_mb,
            CSeq p_rack_cseq
        ) runs on SipComponent  {
            var integer v_responseNum := 1;
            var template(value) PRACK_Request v_prackReq;

            f_setHeadersGeneral(vc_cSeq, "PRACK"); // cseq, contact, branch, via
            if (isvalue(vc_response.msgHeader.rSeq.responseNum)) {
                v_responseNum := vc_response.msgHeader.rSeq.responseNum;
            }
            vc_rAck := m_RAck(v_responseNum, p_rack_cseq.seqNumber, p_rack_cseq.method);

            v_prackReq := m_PRACK_Request_sdp(vc_requestUri, vc_callId, vc_cSeq, vc_from, vc_to, vc_via, vc_rAck, p_mb);

            // added route header if required
            if (isvalue(vc_response.msgHeader.recordRoute)) {
                v_prackReq.msgHeader.route := f_route();
            }

            SIPP.send(v_prackReq) to vc_sent_label;
        }

    }
    group globalSteps {

        //NOTE STF471: removed function f_init_userprofile, have to move up to specific ATS

        /**
         * @desc Initializes the To header
         * @param p_user Identifies the selected user's SIP URL
         * @return The To header
         */
        function f_initToHeader(
            in SipUrl p_user
        ) runs on ImsComponent 
        return To  {
            var To v_to := c_empty_To;
            v_to := {
                fieldName := TO_E,
                addressField := {
                    nameAddr := {
                        displayName := omit,  // optional charstring
                        addrSpec := p_user // SipUrl
                    }
                },  // end addressField
                toParams := omit
            };
            return (v_to);
        }

        /**
         * @desc Initializes the From header
         * @param p_user Identifies the selected user' SIP URL
         * @param p_tag_str The string value of the tag id
         * @return The From header
         */
        function f_initFromHeader(
            in SipUrl p_user,
            charstring p_tag_str
        ) runs on ImsComponent 
        return From  {
            var From v_from := c_empty_From;
            v_from := {
                fieldName := FROM_E,
                addressField := {
                    nameAddr := {
                        displayName := omit,  // optional charstring
                        addrSpec := p_user // SipUrl
                    }
                },  // end addressField
                fromParams := {
                    {
                        id := c_tagId,
                        paramValue := {
                            tokenOrHost := p_tag_str
                        }
                    }
                }
            };
            return v_from;
        }

        //NOTE STF471: removed function f_initSipUrl, have to move up to specific ATS

        //NOTE STF471: removed function f_init_interfaceprofile, have to move up to specific ATS

        /**
         * @desc Retrieves user DomainPort value of the user profile
         * @param p_userProfile Identifies the selected user profile
         * @return The Domain HostPort
         */
        function f_get_userDomainPort(
            in SipUserProfile p_userProfile
        ) runs on ImsComponent
        return HostPort  {
            return
                {
                    p_userProfile.homeDomain,
                    p_userProfile.currPort
                };
        }

        /**
         * @desc Retrieves HostPort value of the interface profile
         * @param p_interface Identifies the selected interface
         * @return The HostPort
         */
        function f_get_interfaceHostPort(
            in ImsInterfaceProfile p_interface
        ) runs on ImsComponent
        return HostPort  {
            return
                {
                    p_interface.SUTIpaddr,
                    p_interface.SUTPort
                };
        }

        /**
         * @desc Retrieves DomainPort value of the interface profile
         * @param p_interface Identifies the selected interface
         * @return The Domain HostPort
         */
        function f_get_interfaceDomainPort(
            in ImsInterfaceProfile p_interface
        ) runs on ImsComponent
        return  HostPort  {
            return
                {
                    p_interface.SUTHomeDomain,
                    p_interface.SUTPort
                };
        }

        /**
         * @desc Gets the receive template for the NameAddr of the interface
         * @param p_interface Identifies the selected interface
         * @return The NameAddr template
         */
        function f_get_interfaceNameAddr(
            in ImsInterfaceProfile p_interface
        ) runs on ImsComponent 
        return template NameAddr  {
            return mw_SipUrl_SUTinterface(p_interface.SUTIpaddr, p_interface.SUTPort);
        }

    }
    
    group registration {

        /**
         * @desc Registration and authentication
         * @param p_cSeq_s The current cseq
         * @param p_auth flag indicating if authentication is needed
         * @param p_host P-CSCF domain name (formerly value from module parameter PX_IMS_TS_PCSCF_HOME_DOMAIN)
         * @param p_port P-CSCF port number (formerly value from module parameter PX_IMS_TS_PCSCF_PORT)
         */
        function f_Registration_IMS(
            inout CSeq p_cSeq_s,
            in boolean p_auth,
            template(value) charstring p_host,
            template(value) integer p_port
        ) runs on SipComponent  {
            f_setHeaders_REGISTER(p_cSeq_s);
            f_SendREGISTER(m_REGISTER_Request_IMS(vc_requestUri, vc_callId, vc_cSeq, vc_from, vc_to, vc_via_REG, vc_contact, m_add_Authorization_digest(vc_authorization, {c_Integrity_protected_no}), p_host, p_port)); // LibSip
            // awaiting of 401 and sending 2nd REGISTER and awaiting 200 OK REGISTER
            if (p_auth) {
                // receiving 401 Unauthorized response.
                // and Re-send REGISTER request with Authorization header
                tc_ack.start;
                alt {
                    [] SIPP.receive(mw_Response_Base(c_statusLine401, vc_callId, p_cSeq_s)) -> value vc_response {
                        tc_ack.stop;
                        // set headers via, cseq and authorization
                        f_setHeaders_2ndREGISTER(p_cSeq_s);
                        f_SendREGISTER(m_REGISTER_Request_IMS(vc_requestUri, vc_callId, vc_cSeq, vc_from, vc_to, vc_via_REG, vc_contact, m_add_Authorization_digest(vc_authorization, {c_Integrity_protected_yes}), p_host, p_port)); // LibSip
                        // awaiting 200 OK REGISTER
                        f_awaitingOkResponse(p_cSeq_s);
                        f_getServiceRouteMapIntoRouteInRegistration(vc_response);
                    }
                    [] SIPP.receive(mw_Response_Base(c_statusLine200, vc_callId, p_cSeq_s)) {
                        tc_ack.stop;
                        log("*** " & __SCOPE__ & ": INCONC: Authorization was not requested as expected ***");
                        setverdict(inconc);
                    }
                }
            }
            else {
                f_awaitingOkResponse(p_cSeq_s);
                f_getServiceRouteMapIntoRouteInRegistration(vc_response);
            }
        } // end function f_Registration_IMS

        /**
         * @desc awaiting UE1 registration (with authentication)
         * @param p_cSeq_s The current cseq
         * @param p_to_user The selected user's SIP URL
         * @param p_qop of the peer UE (alternatively)
         * @param p_register The expected REGISTER_Request - default: mw_REGISTER_authorizedRequest_IMS
         */
        function f_awaitingRegistration_IMS(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in charstring p_qop,
            in template(present) REGISTER_Request p_register := 
                mw_REGISTER_authorizedRequest_IMS(
                    mw_Authorization_digest(
                        mw_digestResponse(
                            (c_Integrity_protected_yes, c_Integrity_protected_ip_assoc_pending)
                        )
                    )
                )
        ) runs on ImsComponent  {
            var SipUrl v_passociated_url;
            
            // Awaiting 1st REGISTER
            f_awaitingREGISTER(mw_REGISTER_unauthorizedRequest_IMS);

            // 401 response
            f_sendResponse(
                m_Response_WWWauthenticate_IMS(
                    c_statusLine401, 
                    vc_callId, 
                    p_cSeq_s, 
                    vc_caller_From, 
                    vc_caller_To, 
                    vc_via, 
                    m_wwwAuthenticate(
                        f_calculatecChallenge_forWWWAuthorization(
                            p_qop, 
                            valueof(vc_authorization.body[0].digestResponse)
                        )
                    )
                )
            );

            // Awaiting 2nd REGISTER with authorization header
            f_awaitingREGISTER(p_register);

            // 200OK to complete the request
            if (isvalue(vc_request.msgHeader.contact)) {
                vc_contact := valueof(vc_request.msgHeader.contact);
                vc_contact.contactBody.contactAddresses[0].contactParams := {
                    {
                        "expires",
                        {
                            tokenOrHost := int2str(3600) 
                        }
                    }
                };
            }
            v_passociated_url := vc_caller_From.addressField.nameAddr.addrSpec;
            f_sendResponse(m_Response_2xxonREGISTER_IMS(c_statusLine200, vc_callId, p_cSeq_s, vc_callee_From, vc_callee_To, vc_via, vc_contact, p_to_user, v_passociated_url));
        } // end function f_awaitingRegistration_IMS

        /**
         * @desc Awaiting UE1 registration (with authentication, without security headers)
         * @param p_cSeq_s The current cseq
         * @param p_to_user The selected user's SIP URL
         * @param p_qop of the peer UE (alternatively)
         */
        function f_awaitingRegistration_IMS_gm(
            inout CSeq p_cSeq_s,
            in SipUrl p_to_user,
            in charstring p_qop
        ) runs on ImsComponent  {
            vc_ignore_subscribe := true; // handle SUBSCRIBE during the call
            
            f_awaitingRegistration_IMS(p_cSeq_s, p_to_user, p_qop, mw_REGISTER_authorizedRequest_wo_securityheaders_IMS);
            
            vc_DeregDone := false;
        } // end function f_awaitingRegistration_IMS_gm

        /**
         * @desc Remove registration
         * @param p_cSeq_s The current cseq
         * @param p_host P-CSCF domain name (formerly value from module parameter PX_IMS_TS_PCSCF_HOME_DOMAIN)
         * @param p_port P-CSCF port number (formerly value from module parameter PX_IMS_TS_PCSCF_PORT)
         */
        function f_removeRegistration_IMS(
            inout CSeq p_cSeq_s,
            template(value) charstring p_host,
            template(value) integer p_port
        ) runs on SipComponent  {
            var template(value) REGISTER_Request v_request;

            if (PX_SIP_REGISTRATION) {
                f_setHeaders_deREGISTER(p_cSeq_s);
                v_request := m_REGISTER_Request_expires_IMS(vc_requestUri, vc_callIdReg, p_cSeq_s, vc_from, vc_to, vc_via, vc_contact, vc_authorization, p_host, p_port, "0");
                v_request.msgHeader.route := f_route();

                f_SendREGISTER(v_request);
                if (PX_SIP_REGISTER_AUTHENTICATION_ENABLED) {
                    // receiving 401 Unauthorized response.
                    // and Re-send REGISTER request with Authorization header
                    tc_ack.start;
                    alt {
                        [] SIPP.receive(mw_Response_Base(c_statusLine401, vc_callIdReg, p_cSeq_s)) -> value vc_response {
                            tc_ack.stop;
                            // set headers via, cseq and authorization
                            f_setHeaders_2ndREGISTER(p_cSeq_s);
                            v_request := m_REGISTER_Request_expires_IMS(vc_requestUri, vc_callIdReg, p_cSeq_s, vc_from, vc_to, vc_via, vc_contact, vc_authorization, p_host, p_port, "0");
                            v_request.msgHeader.route := f_route();
                            // Re-send protected REGISTER
                            f_SendREGISTER(v_request); // LibSip
                            // awaiting 200 OK REGISTER
                            f_awaitingResponse(mw_Response_Base(c_statusLine200, vc_callIdReg, p_cSeq_s));
                        }
                        [] SIPP.receive(mw_Response_Base(c_statusLine200, vc_callIdReg, p_cSeq_s)) -> value vc_response {
                            tc_ack.stop;
                            f_setHeadersOnReceiptOfResponse(vc_response);
                            log("*** " & __SCOPE__ & ": INFO: Authorization was not requested as expected ***");
                        }
                    }
                }
                else {
                    f_awaitingResponse(mw_Response_Base(c_statusLine200, vc_callIdReg, p_cSeq_s));
                }
            }
        } // end f_RemoveRegistration_IMS

    } // end group registration

    group preambles {

        /**
         * @desc Sets variables and default initialization for user profile where proxy role is present
         * @param p_userprofile User profile of call
         * @param p_cSeq_s The current cseq
         */
        function f_IMS_preamble_MGCF(
            inout CSeq p_cSeq_s
        ) runs on ImsComponent  {
            vc_boo_recordRoute := true;
            vc_boo_route := true;
            //NOTE STF471: removed f_init_interfaceprofile(c_interfaceProfile_IMS_SUT_MGCF);

            f_IMS_preamble_woRegistration(p_cSeq_s);
        }

        /**
         * @desc Sets variables and default initialization for user profile
         * @param p_cSeq_s The current cseq
         */
        function f_IMS_preamble_woRegistration(
            inout CSeq p_cSeq_s
        ) runs on ImsComponent  {
            // avoid deregistration in default behavior
            vc_DeregDone := true;
            vc_boo_route := true;
            vc_boo_recordRoute := true;

            // Variables & defaults initialization
            LibSip_Steps.f_init_component(p_cSeq_s);

            // Preamble
            //NOTE STF471: removed f_init_userprofile(p_userprofile); // assignment of PIXIT values to component variable
            vc_sdp_local := valueof(m_SDP_bandwidth(m_media_dynPT(PX_SIP_SDP_DYN, PX_SIP_SDP_ENCODING, PX_SIP_SDP_CLOCKRATE, omit), vc_userprofile));
            vc_contact := m_Contact(m_SipUrl_contactIpaddr(vc_userprofile));
        }

        //NOTE STF471 added
        /**
         * @desc Sets variables and default initialization for user profile
         * @param p_cSeq_s The current cseq
         */
        function f_IMS_preamble_woRegistration_as(
            inout CSeq p_cSeq_s
        ) runs on ImsComponent  {
            // avoid deregistration in default behavior
            vc_DeregDone := true;
            vc_boo_route := true;
            vc_boo_recordRoute := true;

            // Variables & defaults initialization
            LibSip_Steps.f_init_component(p_cSeq_s);

            // Preamble
            //NOTE STF471: removed f_init_userprofile(p_userprofile); // assignment of PIXIT values to component variable
            vc_sdp_local := valueof(m_SDP_bandwidth(m_media_dynPT(PX_SIP_SDP_DYN, PX_SIP_SDP_ENCODING, PX_SIP_SDP_CLOCKRATE, omit), vc_userprofile));
        }

        /**
         * @desc Sets variables and default initialization for user profile and handle registration and authentication with MD5
         * @param p_cSeq_s The current cseq
         * @param p_register Register template
         */
        function f_IMS_preamble_withEmergencyRegistration(
            inout CSeq p_cSeq_s,
            template REGISTER_Request p_register
        ) runs on ImsComponent  {

            //NOTE STF471: removed f_init_interfaceprofile(p_interface);
            // Variables & defaults initialization
            f_IMS_preamble_woRegistration(p_cSeq_s);
            // Preamble
            f_Registration(p_cSeq_s, p_register, PX_SIP_REGISTER_AUTHENTICATION_ENABLED, true);

            // deregistration in case of successful registration
            vc_DeregDone := false;
        }

        /**
         * @desc Sets variables and default initialization for user profile and handle registration and authentication with MD5
         * @param p_cSeq_s The current cseq
         * @param p_register Register template
         */
        function f_IMS_preamble_withRegistration(
            inout CSeq p_cSeq_s,
            out template REGISTER_Request p_register
        ) runs on ImsComponent  {

            //NOTE STF471: removed f_init_interfaceprofile(p_interface);
            // Variables & defaults initialization
            f_IMS_preamble_woRegistration(p_cSeq_s);

            // Preamble
            f_Registration(p_cSeq_s, p_register, PX_SIP_REGISTER_AUTHENTICATION_ENABLED);

            // deregistration in case of successful registration
            vc_DeregDone := false;