📄 t_fifo.c
字号:
return 0;}int init_twrite_lines(){ int i; /* init the line table */ for(i=0;i<TWRITE_PARAMS;i++) { iov_lines_eol[2*i].iov_base = 0; iov_lines_eol[2*i].iov_len = 0; iov_lines_eol[2*i+1] = eol; } /* first line is the version - fill it now */ eol_line_s(0) = TWRITE_VERSION_S; eol_line_len(0) = TWRITE_VERSION_LEN; return 0;}static int inline write_to_fifo(char *fifo, int cnt ){ int fd_fifo; /* open FIFO file stream */ if((fd_fifo = open(fifo,O_WRONLY | O_NONBLOCK)) == -1){ switch(errno){ case ENXIO: LOG(L_ERR,"ERROR:tm:write_to_fifo: nobody listening on " " [%s] fifo for reading!\n",fifo); default: LOG(L_ERR,"ERROR:tm:write_to_fifo: failed to open [%s] " "fifo : %s\n", fifo, strerror(errno)); } goto error; } /* write now (unbuffered straight-down write) */repeat: if (writev(fd_fifo, iov_lines_eol, 2*cnt)<0) { if (errno!=EINTR) { LOG(L_ERR, "ERROR:tm:write_to_fifo: writev failed: %s\n", strerror(errno)); close(fd_fifo); goto error; } else { goto repeat; } } close(fd_fifo); DBG("DEBUG:tm:write_to_fifo: write completed\n"); return 1; /* OK */error: return -1;}static inline char* add2buf(char *buf, char *end, char *title, int title_len, char *value , int value_len){ if (buf+title_len+value_len+2+1>=end) return 0; memcpy( buf, title, title_len); buf += title_len; *(buf++) = ':'; *(buf++) = ' '; memcpy( buf, value, value_len); buf += value_len; *(buf++) = '\n'; return buf;}static inline char* append2buf( char *buf, int len, struct sip_msg *req, struct hdr_avp *ha){ struct hdr_field *hdr; struct usr_avp *avp; int_str avp_val; int_str avp_name; char *end; str foo; int msg_parsed; end = buf+len; msg_parsed = 0; while(ha) { if (ha->type==ELEM_IS_AVP) { /* search for the AVP */ if (ha->sval.s) { avp_name.s=&ha->sval; avp = search_first_avp( AVP_NAME_STR, avp_name, &avp_val); DBG("AVP <%.*s>: %p\n",avp_name.s->len,avp_name.s->s, avp); } else { avp_name.n=ha->ival; avp = search_first_avp( 0, avp_name, &avp_val); DBG("AVP <%i>: %p\n",avp_name.n, avp); } if (avp) { if (avp->flags&AVP_VAL_STR) { buf=add2buf( buf, end, ha->title.s, ha->title.len, avp_val.s->s , avp_val.s->len); if (!buf) goto overflow_err; } else { foo.s=int2str( (unsigned long)avp_val.n, &foo.len); buf=add2buf( buf, end, ha->title.s, ha->title.len, foo.s , foo.len); if (!buf) goto overflow_err; } } } else if (ha->type==ELEM_IS_HDR) { /* parse the HDRs */ if (!msg_parsed) { if (parse_headers( req, HDR_EOH, 0)!=0) { LOG(L_ERR,"ERROR:tm:append2buf: parsing hdrs failed\n"); goto error; } msg_parsed = 1; } /* search the HDR */ if (ha->ival==HDR_OTHER) { for(hdr=req->headers;hdr;hdr=hdr->next) if (ha->sval.len==hdr->name.len && strncasecmp( ha->sval.s, hdr->name.s, hdr->name.len)==0) break; } else { for(hdr=req->headers;hdr;hdr=hdr->next) if (ha->ival==hdr->type) break; } if (hdr) { trim_len( foo.len, foo.s, hdr->body); buf=add2buf( buf, end, ha->title.s, ha->title.len, foo.s , foo.len); if (!buf) goto overflow_err; } } else { LOG(L_ERR,"BUG:tm:append2buf: unknown element type %d\n", ha->type); goto error; } ha = ha->next; } return buf;overflow_err: LOG(L_ERR,"ERROR:tm:append2buf: overflow -> append exceeded %d len\n",len);error: return 0;}static int assemble_msg(struct sip_msg* msg, struct tw_info *twi){ static char id_buf[IDBUF_LEN]; static char route_buffer[ROUTE_BUFFER_MAX]; static char append_buf[APPEND_BUFFER_MAX]; static char cmd_buf[CMD_BUFFER_MAX]; static str empty_param = {".",1}; unsigned int hash_index, label; contact_body_t* cb=0; contact_t* c=0; name_addr_t na; rr_t* record_route; struct hdr_field* p_hdr; param_hooks_t hooks; int l; char* s, fproxy_lr; str route, next_hop, append, tmp_s, body, str_uri; if(msg->first_line.type != SIP_REQUEST){ LOG(L_ERR,"assemble_msg: called for something else then" "a SIP request\n"); goto error; } /* parse all -- we will need every header field for a UAS */ if ( parse_headers(msg, HDR_EOH, 0)==-1) { LOG(L_ERR,"assemble_msg: parse_headers failed\n"); goto error; } /* find index and hash; (the transaction can be safely used due * to refcounting till script completes) */ if( t_get_trans_ident(msg,&hash_index,&label) == -1 ) { LOG(L_ERR,"assemble_msg: t_get_trans_ident failed\n"); goto error; } /* parse from header */ if (msg->from->parsed==0 && parse_from_header(msg)==-1 ) { LOG(L_ERR,"assemble_msg: while parsing <From:> header\n"); goto error; } /* parse the RURI (doesn't make any malloc) */ msg->parsed_uri_ok = 0; /* force parsing */ if (parse_sip_msg_uri(msg)<0) { LOG(L_ERR,"assemble_msg: uri has not been parsed\n"); goto error; } /* parse contact header */ str_uri.s = 0; str_uri.len = 0; if(msg->contact) { if (msg->contact->parsed==0 && parse_contact(msg->contact)==-1) { LOG(L_ERR,"assemble_msg: error while parsing " "<Contact:> header\n"); goto error; } cb = (contact_body_t*)msg->contact->parsed; if(cb && (c=cb->contacts)) { str_uri = c->uri; if (find_not_quoted(&str_uri,'<')) { parse_nameaddr(&str_uri,&na); str_uri = na.uri; } } } /* str_uri is taken from caller's contact or from header * for backwards compatibility with pre-3261 (from is already parsed)*/ if(!str_uri.len || !str_uri.s) str_uri = get_from(msg)->uri; /* parse Record-Route headers */ route.s = s = route_buffer; route.len = 0; fproxy_lr = 0; next_hop = empty_param; p_hdr = msg->record_route; if(p_hdr) { if (p_hdr->parsed==0 && parse_rr(p_hdr)!=0 ) { LOG(L_ERR,"assemble_msg: while parsing " "'Record-Route:' header\n"); goto error; } record_route = (rr_t*)p_hdr->parsed; } else { record_route = 0; } if( record_route ) { if ( (tmp_s.s=find_not_quoted(&record_route->nameaddr.uri,';'))!=0 && tmp_s.s+1!=record_route->nameaddr.uri.s+ record_route->nameaddr.uri.len) { /* Parse all parameters */ tmp_s.len = record_route->nameaddr.uri.len - (tmp_s.s- record_route->nameaddr.uri.s); if (parse_params( &tmp_s, CLASS_URI, &hooks, &record_route->params) < 0) { LOG(L_ERR,"assemble_msg: error while parsing " "record route uri params\n"); goto error; } fproxy_lr = (hooks.uri.lr != 0); DBG("assemble_msg: record_route->nameaddr.uri: %.*s\n", record_route->nameaddr.uri.len,record_route->nameaddr.uri.s); if(fproxy_lr){ DBG("assemble_msg: first proxy has loose routing.\n"); copy_route(s,route.len,record_route->nameaddr.uri.s, record_route->nameaddr.uri.len); } } for(p_hdr = p_hdr->next;p_hdr;p_hdr = p_hdr->next) { /* filter out non-RR hdr and empty hdrs */ if( (p_hdr->type!=HDR_RECORDROUTE) || p_hdr->body.len==0) continue; if(p_hdr->parsed==0 && parse_rr(p_hdr)!=0 ){ LOG(L_ERR,"assemble_msg: " "while parsing <Record-route:> header\n"); goto error; } for(record_route=p_hdr->parsed; record_route; record_route=record_route->next){ DBG("assemble_msg: record_route->nameaddr.uri: " "<%.*s>\n", record_route->nameaddr.uri.len, record_route->nameaddr.uri.s); copy_route(s,route.len,record_route->nameaddr.uri.s, record_route->nameaddr.uri.len); } } if(!fproxy_lr){ copy_route(s,route.len,str_uri.s,str_uri.len); str_uri = ((rr_t*)msg->record_route->parsed)->nameaddr.uri; } else { next_hop = ((rr_t*)msg->record_route->parsed)->nameaddr.uri; } } DBG("assemble_msg: calculated route: %.*s\n", route.len,route.len ? route.s : ""); DBG("assemble_msg: next r-uri: %.*s\n", str_uri.len,str_uri.len ? str_uri.s : ""); if ( REQ_LINE(msg).method_value==METHOD_INVITE || (twi->append && twi->append->add_body) ) { /* get body */ if( (body.s = get_body(msg)) == 0 ){ LOG(L_ERR, "assemble_msg: get_body failed\n"); goto error; } body.len = msg->len - (body.s - msg->buf); } else { body = empty_param; } /* additional headers */ append.s = s = append_buf; if (sizeof(flag_t)*2+12+1 >= APPEND_BUFFER_MAX) { LOG(L_ERR,"assemble_msg: buffer overflow " "while copying optional header\n"); goto error; } append_str(s,"P-MsgFlags: ",12); l = APPEND_BUFFER_MAX - (12+1); /* include trailing `\n'*/ if (int2reverse_hex(&s, &l, (int)msg->msg_flags) == -1) { LOG(L_ERR,"assemble_msg: buffer overflow " "while copying optional header\n"); goto error; } append_chr(s,'\n'); if ( twi->append && ((s=append2buf( s, APPEND_BUFFER_MAX-(s-append.s), msg, twi->append->elems))==0) ) goto error; /* body separator */ append_chr(s,'.'); append.len = s-append.s; eol_line_s(1) = s = cmd_buf; if(twi->action.len+12 >= CMD_BUFFER_MAX){ LOG(L_ERR,"assemble_msg: buffer overflow while " "copying command name\n"); goto error; } append_str(s,"sip_request.",12); append_str(s,twi->action.s,twi->action.len); eol_line_len(1) = s - (char*)eol_line_s(1); eol_line(2,REQ_LINE(msg).method); /* method type */ eol_line(3,msg->parsed_uri.user); /* user from r-uri */ eol_line(4,msg->parsed_uri.host); /* domain */ eol_line(5,msg->rcv.bind_address->address_str); /* dst ip */ eol_line(6,msg->rcv.dst_port==SIP_PORT ? empty_param : msg->rcv.bind_address->port_no_str); /* port */ /* r_uri ('Contact:' for next requests) */ eol_line(7,msg->first_line.u.request.uri); /* r_uri for subsequent requests */ eol_line(8,str_uri.len?str_uri:empty_param); eol_line(9,get_from(msg)->body); /* from */ eol_line(10,msg->to->body); /* to */ eol_line(11,msg->callid->body); /* callid */ eol_line(12,get_from(msg)->tag_value); /* from tag */ eol_line(13,get_to(msg)->tag_value); /* to tag */ eol_line(14,get_cseq(msg)->number); /* cseq number */ eol_line_s(15)=id_buf; /* hash:label */ s = int2str(hash_index, &l); if (l+1>=IDBUF_LEN) { LOG(L_ERR, "assemble_msg: too big hash\n"); goto error; } memcpy(id_buf, s, l); id_buf[l]=':'; eol_line_len(15)=l+1; s = int2str(label, &l); if (l+1+eol_line_len(15)>=IDBUF_LEN) { LOG(L_ERR, "assemble_msg: too big label\n"); goto error; } memcpy(id_buf+eol_line_len(15), s, l); eol_line_len(15) += l; eol_line(16, route.len ? route : empty_param ); eol_line(17, next_hop ); eol_line(18, append ); eol_line(19, body ); /* success */ return 1;error: /* 0 would lead to immediate script exit -- -1 returns * with 'false' to script processing */ return -1;}static int write_to_unixsock(char* sockname, int cnt){ int len, e; struct sockaddr_un dest; if (!sockname) { LOG(L_ERR, "write_to_unixsock: Invalid parameter\n"); return E_UNSPEC; } len = strlen(sockname); if (len == 0) { DBG("write_to_unixsock: Error - empty socket name\n"); return -1; } else if (len > 107) { LOG(L_ERR, "write_to_unixsock: Socket name too long\n"); return -1; } memset(&dest, 0, sizeof(dest)); dest.sun_family = PF_LOCAL; memcpy(dest.sun_path, sockname, len);#ifdef HAVE_SOCKADDR_SA_LEN dest.sun_len = len;#endif e = connect(sock, (struct sockaddr*)&dest, SUN_LEN(&dest));#ifdef HAVE_CONNECT_ECONNRESET_BUG /* * Workaround for a nasty bug in BSD kernels dated back * to the Berkeley days, so that can be found in many modern * BSD-derived kernels. Workaround should be pretty harmless since * in normal conditions connect(2) can never return ECONNRESET. */ if ((e == -1) && (errno == ECONNRESET)) e = 0;#endif if (e == -1) { LOG(L_ERR, "write_to_unixsock: Error in connect: %s\n", strerror(errno)); return -1; } if (tsend_dgram_ev(sock, iov_lines_eol, 2 * cnt, tm_unix_tx_timeout * 1000) < 0) { LOG(L_ERR, "write_to_unixsock: writev failed: %s\n", strerror(errno)); return -1; } return 0;}int t_write_req(struct sip_msg* msg, char* vm_fifo, char* info){ if (assemble_msg(msg, (struct tw_info*)info) < 0) { LOG(L_ERR, "ERROR:tm:t_write_req: Error int assemble_msg\n"); return -1; } if (write_to_fifo(vm_fifo, TWRITE_PARAMS) == -1) { LOG(L_ERR, "ERROR:tm:t_write_req: write_to_fifo failed\n"); return -1; } /* make sure that if voicemail does not initiate a reply * timely, a SIP timeout will be sent out */ if (add_blind_uac() == -1) { LOG(L_ERR, "ERROR:tm:t_write_req: add_blind failed\n"); return -1; } return 1;}int t_write_unix(struct sip_msg* msg, char* socket, char* info){ if (assemble_msg(msg, (struct tw_info*)info) < 0) { LOG(L_ERR, "ERROR:tm:t_write_unix: Error in assemble_msg\n"); return -1; } if (write_to_unixsock(socket, TWRITE_PARAMS) == -1) { LOG(L_ERR, "ERROR:tm:t_write_unix: write_to_unixsock failed\n"); return -1; } /* make sure that if voicemail does not initiate a reply * timely, a SIP timeout will be sent out */ if (add_blind_uac() == -1) { LOG(L_ERR, "ERROR:tm:t_write_unix: add_blind failed\n"); return -1; } return 1;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -