aboutsummaryrefslogblamecommitdiffstats
path: root/lib/dialyzer/test/user_SUITE_data/src/wsp_pdu.erl
blob: e5b387478f73b4eadbe7907b96fc23bdf1182940 (plain) (tree)
3909
3910
3911
3912
3913
3914
3915
3916
3917
3918
3919
3920
3921
3922
3923
3924
3925
3926
3927
3928
3929
3930
3931
3932
3933
3934
3935
3936
3937
3938
3939
3940
3941
3942
3943
3944
3945
3946
3947
3948
3949
3950
3951
3952
3953
3954
3955
3956
3957
3958
3959
3960
3961
3962
3963
3964
3965
3966
3967
3968
3969
3970
3971
3972
3973
3974
3975
3976
3977
3978
3979
3980
3981
3982
3983
3984
3985
3986
3987
3988
3989
3990
3991
3992
3993
3994
3995
3996
3997
3998
3999
4000
4001
4002
4003
4004
4005
4006
4007
4008
4009
4010
4011
4012
4013
4014
4015
4016
4017
4018
4019
4020
4021
4022
4023
4024
4025
4026
4027
4028
4029
4030
4031
4032
4033
4034
4035
4036
4037
4038
4039
4040
4041
4042
4043
4044
4045
4046
4047
4048
4049
4050
4051
4052
4053
4054
4055
4056
4057
4058
4059
4060
4061
4062
4063
4064
4065
4066
4067
4068
4069
4070
4071
4072
4073
4074
4075
4076
4077
4078
4079
4080
4081
4082
4083
4084
4085
4086
4087
4088
4089
4090
4091
4092
4093
4094
4095
4096
4097
4098
4099
4100
4101
4102
4103
4104
4105
4106
4107
4108
4109
4110
4111
4112
4113
4114
4115
4116
4117
4118
4119
4120
4121
4122
4123
4124
4125
4126
4127
4128
4129
4130
4131
4132
4133
4134
4135
4136
4137
4138
4139
4140
4141
4142
4143
4144
4145
4146
4147
4148
4149
4150
4151
4152
4153
4154
4155
4156
4157
4158
4159
4160
4161
4162
4163
4164
4165
4166
4167
4168
4169
4170
4171
4172
4173
4174
4175
4176
4177
4178
4179
4180
4181
4182
4183
4184
4185
4186
4187
4188
4189
4190
4191
4192
4193
4194
4195
4196
4197
4198
4199
4200
4201
4202
4203
4204
4205
4206
4207
4208
4209
4210
4211
4212
4213
4214
4215
4216
4217
4218
4219
4220
4221
4222
4223
4224
4225
4226
4227
4228
4229
4230
4231
4232
4233
4234
4235
4236
4237
4238
4239
4240
4241
4242
4243
4244
4245
4246
4247
4248
4249
4250
4251
4252
4253
4254
4255
4256
4257
4258
4259
4260
4261
4262
4263
4264
4265
4266
4267
4268
4269
4270
4271
4272
4273
4274
4275
4276
4277
4278
4279
4280
4281
4282
4283
4284
4285
4286
4287
4288
4289
4290
4291
4292
4293
4294
4295
4296
4297
4298
4299
4300
4301
4302
4303
4304
4305
4306
4307
4308
4309
4310
4311
4312
4313
4314
4315
4316
4317
4318
4319
4320
4321
4322
4323
4324
4325
4326
4327
4328
4329
4330
4331
4332
4333
4334
4335
4336
4337
4338






















                                                                          





                                                                    
















                                                        
 



















































                                                                          
 







































                                                                      
                                                           







                                          
 




























                                                                      
                                                 



























































































































                                                                             
                                                                  





                                                                      
 



















































                                                                        
 

































                                                           
 





                                                  
                                            








































                                                                          
 







































                                                                           
 


















































                                                                       
                                                          


                                    
                  




















































                                                                              
                       

























                                                                              
                                 

















                                                                        
                                                       
























                                                                               
                          



















                                                      
                               









                                                   
                           

                                                  
                                                    
                                                   
                            



                                                   
                          
                                                 
                
                                      
                  
                                        
                          

                                                
                                                  


                                                  
                          

                                                
                       
                                             
                         
                                               
                             
                                                   
                             
                                                   
                           
                                                 
                             
                                                   
                        

                                              
                                                  
                                                
                          

                                                
                         
                                               
                 
                                        
                 
                                       
                    


                                          
                 
                                       
                              
                                                    
                     
                                           
                          
                                                
                     
                                           
                                
                                                      
                     
                                           
                          
                                                
                         
                                               
                   
                                         
                               
                                                     
                                
                                                      
                   
                                         
                  
                                        
                    
                                          
                        
                                              
                   
                                         
                              
                                                    
                    
                                          
                       
                                             
                 
                                       
                
                                      
                    
                                          
                             



                                                        
                                

                                                      
                                                         

                                                       
                                                      

                                                    
                                                        

                                                      
                                                       

                                                     
                                                      

                                                    
                                              

                                            
                                            

                                          
                                                 

                                               
                                                    

                                                  
                                           
                                         
                                           

                                         
                                       
                                     
                                            

                                          
                                              
                                            
                                              

                                            
                                               
                                             
                                               
                                             
                                           



                                                     
                                                    

                                                  
                                                   


                                                              
                                                       









                                                                              
 








                                                              
                                                   
                                                       
                           
                                                       
                                                    
                                                        
                            


                                                        
                          
                                                      
                                                       
                                                          
                
                                            
                  
                                              
                          
                                                      
                                                  




                                                      
                       
                                                   
                         
                                                     
                             

                                                         
                             
                                                        
                           
                                                      
                             
                                                        
                        
                                                   
                                                  
                                                     
                          
                                                     
                         
                                                    
                 
                                            
                 
                                            
                    


                                               
                 
                                            
                              
                                                         
                     
                                                
                          
                                                     
                     
                                                
                                
                                                           
                     
                                                
                          
                                                     
                         
                                                    
                   
                                              
                               
                                                          
                                
                                                           
                   
                                              
                  
                                             
                    
                                               
                        
                                                   
                   
                                              
                              
                                                         
                    
                                               
                       
                                                  
                 
                                            
                
                                           
                    
                                               
                             







                                                           
                                                         
                                                            
                                                      
                                                         
                                                        

                                                           
                                                      
                                                         
                                              

                                                 
                                            
                                               
                                                 
                                                    
                                                    
                                                       
                                                    

                                                       
                                           
                                              
                                           
                                              
                                       
                                          
                                            
                                               
                                              
                                                 
                                               
                                                  
                                               
                                                  
                                           






                                                                                    
                                                   


                                                            
                                                       
                                                          
                                 

                                                     
                                 
















































                                                                  
 






































































                                                                    
            
















                                                                       
                                               









                                                                    
 







































































                                                                 
                 

















                                                                              
  




















































                                                                              
                                     



























































































                                                                                  
 


























                                                              
            



























                                                                              
                                    
                                     
                                    
































































































                                                                              
 













                                                
                                             
                                                         
                                    





                                                  
                                             





































































                                                                              
                                                      





                                                                              
       






                                                                              
                                   

                                         
                                                        



                                                                              
                    









                                                                              
 







                                          
                                       
                                        
                                       












                                                                              
                                




                                       
                                       


































                                                                              
                                                             











                                                                              
                                       



                             
                                       















                                                                              
 

                                  
 






























                                                                              
                                            
                                   
         






                                          
 


























                                                                              

                                                         
                                                                    
                                                      



























                                                                              

                           

















                                                                              
                           

                                           
                               















                                                                              

                              








                                                                              
                







                                                                              
                           

                                           
                               















                                                                              
                           

                                           
                               














                                                                              

                                        


















                                                                              
                               

                                           
                                   














                                                                              
                                    

                                           
                                        
















                                                                              
 






                                        
                               






                                      
                                                    
                                                   
                                   














                                                                              

                                          













                                                                              
                               



                       
                               












                                                                              
                              
                                            
                                       



                                 
                                    





















                                                                              
                                       




















                                                                              
             























                                                                              
                                        




























                                                                              
        














                                                                              
                                  



                                                  
                             





















                                                                              
                            












                                      
                                   






















                                                                              
                              



                       
                              



























                                                                              
                                  






















                                                                              
  











                                                                              
                             


















                                                                              
                                        




                                          
                                        
                                            
                                                            

































                                                                              
                                 

                                           
                                     












                                                                              

                          

                                              
                                                   
                                              
                                                






















                                                                              
                


















                                                                              
                              








                                              
                                                      





















                                                                               
                                        







                                                                              
                








                                                                              

                                         

                              
                                     

                                                                 
                                     
























                                                                              

                                           

                                                
                                               









                                                                              
                                 

                                        
                              

                       
                                        











                                                                              
                                   

                                          
                                

                       
                                          











                                                                              
                                  



                                         

 


































                                                                              
 













                                                                              
  






                                                                              
 




















                                                                              
                              

                                         
                                  



































                                                                              
                            










                                                       
                                      


















                                                                       
                                                              




















                                                                
          

                           
                  























                                                                              

                            



                              
                                            



                                                               
                             
                                      
                                    









                                                                              
  






                                                                              
 






















                                                                
                         


                                          
           



























                                                                              

                             

                                              
                                                      
                                                 
                                                   
















                                                                              
                                

                               
                                    




                                                                              
        








                                                                              
 













































                                                                              
                                        

























































                                                                              
 












                                                                 
               




























                                                                              
                                       
























                                                                              
                           

                       
                                     

                                           
                                         







                                                                              
                                    

                                           
                                 

                       
                                           

                                           
                                               







                                                                              

                                         
 
                               

                       
                                       

                                           
                                             


                                                             
  










































                                                                      
 

















































































































                                                                                  
                                                                                            






























                                                                           
 

























































































































                                                                                   
                                     





















                                                 
 

                              
 

























                                                                          
                          
















































































































                                                                    
 



                                                             
                                         
































































































































                                                          
            































                                                                        

 













































































































































































































































































































































































































































                                             
  
                                                                       
  
































                                                      
 






























                                                                     
                             


























                                                                        
                              

                           
                       




                                                 
 









































































                                                                              
                                      







                                             
                                             


                                                   
 









                                 
        




















                                            
        




































































































                                                                            
  
                                                                            
  





                                                
                                                  
                                    
                                           


















                                                            
                            




                                                          
 






































































                                                                            
                                                           



































                                                                             
                                                                    







                                                                         
 



























































































































                                                                         
 




















































                                                              
   











                                                      
                                            








































                                                                 
                        






















                                                                      
 














































                                                     
          










                                                                  
          








                                                                  
 







                                               
                                                        
                                              
                                                                 
                                                              
                                                                          





                                                                              
                   




































































































                                                                           
                        





                                                        
 








                                                                  
                      


































































































                                                                              
                                

































                                                                          
%%%=======================================================================
%%% File        : wsp_pdu.erl
%%% Author      : Tony Rogvall <[email protected]>
%%% Description : WSP PDU
%%% Created     : 18 Aug 2003 by <[email protected]>
%%%=======================================================================
%%%
%%% There are a couple of bugs in this file. Some are detected by
%%% Dialyzer v1.1 starting both from byte code and from source, some
%%% other ones are detected only starting from sourse, while some
%%% others go unnoticed (these are identified by "BUG" below). It is
%%% expected that at least some of them are detected when the new type
%%% analysis is integrated into Dialyzer. Some other ones, like the
%%% one with the unused _Acc argument are harder to detect and might
%%% require different techniques.
%%%
%%%=======================================================================

-module(wsp_pdu).
-export([encode/1, encode/2, decode/1, decode/2]).

%% The following is just to suppress unused function warnings
-export([decode_address/1, decode_header/2,
	 decode_headers/1, decode_mms_version/1, decode_multipart/1,
	 encode_headers/1, encode_mms_version/1, encode_multipart/1,
	 encode_language/1, encode_short_integer/1,
	 fmt_current_date/0,
	 format_header/1, format_headers/1,
	 parse_header/1, format/1]).

-include("wsp.hrl").
-include("wdp.hrl").

-ifdef(debug).
-define(dbg(Fmt,Args), io:format(Fmt, Args)).
-else.
-define(dbg(Fmt,Args), ok).
-endif.

-define(WARN(Cond, Message),
	if (Cond) ->
		io:format("Warning: ~s\n", [(Message)]);
	   true ->
		ok
	end).


format(Pdu) ->
    if record(Pdu, wsp_connect) ->
	    fmt(Pdu, record_info(fields, wsp_connect));
       record(Pdu, wsp_connect_reply) ->
	    fmt(Pdu, record_info(fields, wsp_connect_reply));
       record(Pdu, wsp_redirect) ->
	    fmt(Pdu, record_info(fields, wsp_redirect));
       record(Pdu, wsp_disconnect) ->
	    fmt(Pdu, record_info(fields, wsp_disconnect));
       record(Pdu, wsp_get) ->
	    fmt(Pdu, record_info(fields, wsp_get));
       record(Pdu, wsp_post) ->
	    fmt(Pdu, record_info(fields, wsp_post));
       record(Pdu,wsp_reply) ->
	    fmt(Pdu, record_info(fields, wsp_reply));
       record(Pdu,wsp_data_fragment_pdu) ->
	    fmt(Pdu, record_info(fields, wsp_data_fragment_pdu));
       record(Pdu,wsp_push) ->
	    fmt(Pdu, record_info(fields, wsp_push));
       record(Pdu, wsp_suspend) ->
	    fmt(Pdu, record_info(fields, wsp_suspend));
       record(Pdu, wsp_resume) ->
	    fmt(Pdu, record_info(fields, wsp_resume));
       record(Pdu, wsp_unknown_pdu) ->
	    fmt(Pdu, record_info(fields, wsp_unknown_pdu))
    end.

fmt(Pdu, Fs) ->
    [Name | Vs] = tuple_to_list(Pdu),
    lists:flatten(["\n",atom_to_list(Name)," {\n" , fmt1(Fs, Vs), "\n}"]).

fmt1([F|Fs],[V|Vs]) ->
    [io_lib:format("  ~s: ~s;\n", [F,fmt_value(V)]) | fmt1(Fs, Vs)];
fmt1([], []) ->
    "".

fmt_value(V) when binary(V) -> "#Bin";
fmt_value(V) -> lists:flatten(io_lib:format("~p",[V])).


%%
%% Wsp pdu encoder
%%
encode(Pdu) ->
    encode(Pdu, ?WSP_DEFAULT_VERSION).

encode(Pdu, Version) ->
    ?dbg("encode pdu using encoding version ~p\n", [Version]),
    Enc = encode1(Pdu, Version),
    ?dbg("pdu: ~p\nreversed pdu: ~p\n",
	 [Pdu, decode(Enc, Version)]),
    Enc.


encode1(Pdu, Version) ->
    case Pdu of
	#wsp_connect_reply {server_session_id=ServerSessionId,
			    capabilities=Capabilities,
			    headers=Headers} ->
	    EncServerSessionId = e_uintvar(ServerSessionId),
	    EncCapabilities = encode_capabilities(Capabilities),
	    EncCapabilitiesLength = e_uintvar(size(EncCapabilities)),
	    EncHeaders = encode_headers(Headers,Version),
	    EncHeadersLength = e_uintvar(size(EncHeaders)),
	    <<?WSP_ConnectReply,
	     EncServerSessionId/binary,
	     EncCapabilitiesLength/binary, EncHeadersLength/binary,
	     EncCapabilities/binary, EncHeaders/binary>>;

	#wsp_reply{ status=Status,
		    content_type=ContentType,
		    headers=Headers,
		    data=Data} ->
	    EncStatus = encode_status_code(Status),
	    EncContentType = encode_content_type(ContentType,Version),
	    EncHeaders = encode_headers(Headers,Version),
	    EncHeadersLength = e_uintvar(size(EncContentType)+
					 size(EncHeaders)),
	    <<?WSP_Reply,
	     EncStatus:8,
	     EncHeadersLength/binary,
	     EncContentType/binary,
	     EncHeaders/binary,
	     Data/binary>>;

	#wsp_post{type=Type, uri=URI, content_type=ContentType,
		  headers=Headers, data=Data} ->
	    %% WSP_Post, WSP_Put
	    PDUType = encode_pdu_type(Type),
	    UriLength = e_uintvar(length(URI)),
	    EncContentType = encode_content_type(ContentType,Version),
	    EncHeaders = encode_headers(Headers,Version),
	    EncHeadersLength = e_uintvar(size(EncContentType)+
					 size(EncHeaders)),
	    %% FIXME
	    <<PDUType:8,
	     UriLength/binary,
	     EncHeadersLength/binary,
	     (list_to_binary(URI))/binary,
	     EncContentType/binary,
	     EncHeaders/binary,
	     Data/binary>>;

	#wsp_push{type=Type, content_type=ContentType,
		  headers=Headers, data=Data} ->
	    %% WSP_Push, WSP_ConfirmedPush
	    PDUType = encode_pdu_type(Type),
	    EncContentType = encode_content_type(ContentType,Version),
	    EncHeaders = encode_headers(Headers,Version),
	    ?dbg("Version ~p Headers ~p", [Version, Headers]),
	    ?dbg("EncHeaders ~p", [EncHeaders]),
	    EncHeadersLength = e_uintvar(size(EncContentType)+
					 size(EncHeaders)),
	    ?dbg("EncCT = ~w ~w", [ContentType, EncContentType]),
	    ?dbg("EncHL = ~w", [EncHeadersLength]),
	    <<PDUType:8,
	     EncHeadersLength/binary,
	     EncContentType/binary,
	     EncHeaders/binary,
	     Data/binary>>;

	#wsp_get{type=Type, uri=URI, headers=Headers} ->
	    %% WSP_Get, WSP_Options, WSP_Head, WSP_Delete, WSP_Trace
	    PDUType = encode_pdu_type(Type),
	    UriLength = length(URI),
	    EncHeaders = encode_headers(Headers,Version),
	    <<PDUType:8,
	     (e_uintvar(UriLength))/binary,
	     (list_to_binary(URI))/binary,
	     EncHeaders/binary>>;

	#wsp_redirect { flags = Flags, addresses = Addrs } ->
	    Flg = lists:foldl(fun(permanent,F) ->
				       ?WSP_PERMANENT_REDIRECT bor F;
			          (resue, F) ->
				       ?WSP_REUSE_SECURITY bor F
			       end, 0, Flags),
	    EncAddr = encode_addresses(Addrs),
	    <<?WSP_Redirect, Flg:8, EncAddr/binary >>;


	#wsp_data_fragment_pdu { headers=Headers, data=Data } ->
	    EncHeaders = encode_headers(Headers,Version),
	    << ?WSP_DataFragmentPDU, EncHeaders/binary, Data/binary >>
    end.

decode(Data) ->
    decode(Data, ?WSP_COMPLIENT_VERSION).

decode(Data0, Version) ->
    case Data0 of
	<<?WSP_Connect:8,PduVersion:8,D0/binary>> ->
	    %% 8.2.2.1
	    {CapabilitiesLen,D1} = d_uintvar(D0),
	    {HeadersLen,D2} = d_uintvar(D1),
	    {Capabilities,D3} = split_binary(D2, CapabilitiesLen),
	    Caps = decode_capabilities(Capabilities,#wsp_capabilities{}),
	    {Headers,D4} = split_binary(D3, HeadersLen),
	    DecHeaders = decode_headers(Headers, Version),
	    ?WARN(D4 =/= <<>>, "Connect pdu contains trailing data"),
	    %% FIXME: warn when D4 is not <<>>
	    #wsp_connect{ version = PduVersion,
			  capabilities=Caps,
			  headers = DecHeaders };

	<<?WSP_ConnectReply:8,D0/binary>> ->
	    %% 8.2.2.2
	    {ServerSessionId,D1} = d_uintvar(D0),
	    {CapabilitiesLen,D2} = d_uintvar(D1),
	    {HeadersLen,D3} = d_uintvar(D2),
	    {Capabilities,D4} = split_binary(D3, CapabilitiesLen),
	    Caps = decode_capabilities(Capabilities,#wsp_capabilities{}),
	    {Headers,D5} = split_binary(D4, HeadersLen),
	    DecHeaders = decode_headers(Headers, Version),
	    ?WARN(D5 =/= <<>>, "ConnectReply pdu contains trailing data"),
	    #wsp_connect_reply{server_session_id=ServerSessionId,
			       capabilities=Caps,
			       headers=DecHeaders};

	<<?WSP_Redirect:8,Flg:8,D0/binary>> ->
	    Flags =
		if Flg band ?WSP_PERMANENT_REDIRECT =/= 0 -> [permanent];
		   true -> []
		end ++
		if Flg band ?WSP_REUSE_SECURITY =/= 0 -> [security];
		   true -> []
		end,
	    Addrs = decode_addresses(D0),
	    %% 8.2.2.3 Redirect
	    #wsp_redirect{flags=Flags,addresses=Addrs};


	<<?WSP_Disconnect:8,D0/binary>> ->
	    %% 8.2.2.4 Disconnect
	    {ServerSessionId,_D1} = d_uintvar(D0),
	    #wsp_disconnect{server_session_id=ServerSessionId};

	<<?WSP_Get:8,D0/binary>> ->
	    {URILength, D1} = d_uintvar(D0),
	    <<UriData:URILength/binary,D2/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_get{type='GET',uri=binary_to_list(UriData),headers=Hs };

	<<?WSP_Options:8,D0/binary>> ->
	    {URILength, D1} = d_uintvar(D0),
	    <<UriData:URILength/binary,D2/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_get{type='OPTIONS',uri=binary_to_list(UriData),headers=Hs };

	<<?WSP_Head:8,D0/binary>> ->
	    {URILength, D1} = d_uintvar(D0),
	    <<UriData:URILength/binary,D2/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_get{type='HEAD',uri=binary_to_list(UriData),headers=Hs };

	<<?WSP_Delete:8,D0/binary>> ->
	    {URILength, D1} = d_uintvar(D0),
	    <<UriData:URILength/binary,D2/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_get{type='DELETE',uri=binary_to_list(UriData),headers=Hs };

	<<?WSP_Trace:8,D0/binary>> ->
	    {URILength, D1} = d_uintvar(D0),
	    <<UriData:URILength/binary,D2/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_get{type='TRACE',uri=binary_to_list(UriData),headers=Hs };

	%% 8.2.3.2 Post
	<<?WSP_Post:8,D0/binary>> ->
	    {URILen, D1} = d_uintvar(D0),
	    {HL0, D2} = d_uintvar(D1),
	    <<UriData:URILen/binary,D3/binary>> = D2,
	    {FieldData,D4} = scan_header_data(D3),
	    HL1 = (HL0-(size(D3)-size(D4))),
	    <<D5:HL1/binary,Data/binary>> = D4,
	    ContentType = decode_content_type(FieldData, Version),
	    Headers = decode_headers(D5, Version),
	    #wsp_post{ type='POST', uri=binary_to_list(UriData),
		       content_type=ContentType, headers=Headers, data=Data};

	<<?WSP_Put:8,D0/binary>> ->
	    {URILen, D1} = d_uintvar(D0),
	    {HL0, D2} = d_uintvar(D1),
	    <<UriData:URILen/binary,D3/binary>> = D2,
	    {FieldData,D4} = scan_header_data(D3),
	    HL1 = (HL0-(size(D3)-size(D4))),
	    <<D5:HL1/binary,Data/binary>> = D4,
	    ContentType = decode_content_type(FieldData, Version),
	    Headers = decode_headers(D5, Version),
	    #wsp_post{ type='PUT', uri=binary_to_list(UriData),
		       content_type=ContentType, headers=Headers, data=Data};

	<<?WSP_Reply:8,StatusCode:8,D0/binary>> ->
	    %% 8.2.3.3 Reply
	    Status = decode_status_code(StatusCode),
	    {HL0, D1} = d_uintvar(D0),
	    {FieldData, D2} = scan_header_data(D1),
	    ContentType = decode_content_type(FieldData, Version),
	    %% Headers are headersLength - binary size of content type
	    HL1 = (HL0-(size(D1)-size(D2))),
	    <<D3:HL1/binary,Data/binary>> = D2,
	    Hs = decode_headers(D3, Version),
	    #wsp_reply{status=Status, content_type=ContentType,
		       headers=Hs, data=Data};

	<<?WSP_DataFragmentPDU:8,D0/binary>> ->
	    %% 8.2.3.4 Data Fragment PDU
	    {HL0, D1} = d_uintvar(D0),
	    <<D2:HL0/binary,Data/binary>> = D1,
	    Hs = decode_headers(D2, Version),
	    #wsp_data_fragment_pdu{headers=Hs, data=Data};

	%% 8.2.4.1 Push or ConfirmedPush
	<<?WSP_Push:8,D0/binary>> ->
	    {HeadersLength, T200} = d_uintvar(D0),
	    {FieldData, T300} = scan_header_data(T200),
	    ContentType = decode_content_type(FieldData, Version),
	    RealHeadersLength = (HeadersLength-(size(T200)-size(T300))),
	    <<T400:RealHeadersLength/binary,Data/binary>> = T300,
	    Headers = decode_headers(T400, Version),
	    #wsp_push{type=push,content_type=ContentType,
		      headers=Headers,data=Data};

	<<?WSP_ConfirmedPush:8,D0/binary>> ->
	    {HeadersLength, T200} = d_uintvar(D0),
	    {FieldData, T300} = scan_header_data(T200),
	    ContentType = decode_content_type(FieldData, Version),
	    RealHeadersLength = (HeadersLength-(size(T200)-size(T300))),
	    <<T400:RealHeadersLength/binary,Data/binary>> = T300,
	    Headers = decode_headers(T400, Version),
	    #wsp_push{type=confirmed_push,
		      content_type=ContentType,
		      headers=Headers,data=Data};

	<<PDUType:8,T100/binary>> ->
	    #wsp_unknown_pdu { type = PDUType, data = T100 }
    end.


encode_pdu_type(connect) -> ?WSP_Connect;
encode_pdu_type(connect_reply) -> ?WSP_ConnectReply;
encode_pdu_type(redirect) -> ?WSP_Redirect;
encode_pdu_type(reply) -> ?WSP_Reply;
encode_pdu_type(disconnect) -> ?WSP_Disconnect;
encode_pdu_type(push) -> ?WSP_Push;
encode_pdu_type(confirmed_push) -> ?WSP_ConfirmedPush;
encode_pdu_type(suspend) -> ?WSP_Suspend;
encode_pdu_type(resume) -> ?WSP_Resume;
encode_pdu_type(data_fragment_pdu) -> ?WSP_DataFragmentPDU;
encode_pdu_type('GET') -> ?WSP_Get;
encode_pdu_type('OPTIONS') -> ?WSP_Options;
encode_pdu_type('HEAD') -> ?WSP_Head;
encode_pdu_type('DELETE') -> ?WSP_Delete;
encode_pdu_type('TRACE') -> ?WSP_Trace;
encode_pdu_type('POST') -> ?WSP_Post;
encode_pdu_type('PUT') -> ?WSP_Put;
encode_pdu_type(Type) when integer(Type) -> Type.


decode_pdu_type(?WSP_Connect) -> connect;
decode_pdu_type(?WSP_ConnectReply) -> connect_reply;
decode_pdu_type(?WSP_Redirect) -> redirect;
decode_pdu_type(?WSP_Reply) -> reply;
decode_pdu_type(?WSP_Disconnect) -> disconnect;
decode_pdu_type(?WSP_Push) -> push;
decode_pdu_type(?WSP_ConfirmedPush) -> confirmed_push;
decode_pdu_type(?WSP_Suspend) -> suspend;
decode_pdu_type(?WSP_Resume) -> resume;
decode_pdu_type(?WSP_DataFragmentPDU) -> data_fragment_pdu;
decode_pdu_type(?WSP_Get) -> 'GET';
decode_pdu_type(?WSP_Options) -> 'OPTIONS';
decode_pdu_type(?WSP_Head) -> 'HEAD';
decode_pdu_type(?WSP_Delete) -> 'DELETE';
decode_pdu_type(?WSP_Trace) -> 'TRACE';
decode_pdu_type(?WSP_Post) -> 'POST';
decode_pdu_type(?WSP_Put) -> 'PUT';
decode_pdu_type(Type) -> Type.  %% allow unknown pdu types.


%% Convert various data types to list

to_list(I) when integer(I) ->
    integer_to_list(I);
to_list(A) when atom(A) ->
    atom_to_list(A);
to_list(Version={X,Y}) when integer(X), integer(Y) ->
    format_version(Version);
to_list(DateTime={{_,_,_},{_,_,_}}) ->
    fmt_date(DateTime);
to_list(L) when list(L) ->
    L.



encode_capabilities(Capa) ->
    encode_capabilities(Capa,#wsp_capabilities{}).

encode_capabilities(Cap,Def) ->
    Known =
	[encode_capability(?WSP_CAP_ALIASES,
			   Cap#wsp_capabilities.aliases,
			   Def#wsp_capabilities.aliases),
	 encode_capability(?WSP_CAP_CLIENT_SDU_SIZE,
			   Cap#wsp_capabilities.client_sdu_size,
			   Def#wsp_capabilities.client_sdu_size),
	 encode_capability(?WSP_CAP_SERVER_SDU_SIZE,
			   Cap#wsp_capabilities.server_sdu_size,
			   Def#wsp_capabilities.server_sdu_size),
	 encode_capability(?WSP_CAP_PROTOCOL_OPTIONS,
			   Cap#wsp_capabilities.protocol_options,
			   Def#wsp_capabilities.protocol_options),
	 encode_capability(?WSP_CAP_METHOD_MOR,
			   Cap#wsp_capabilities.method_mor,
			   Def#wsp_capabilities.method_mor),
	 encode_capability(?WSP_CAP_PUSH_MOR,
			   Cap#wsp_capabilities.push_mor,
			   Def#wsp_capabilities.push_mor),
	 encode_capability(?WSP_CAP_EXTENDED_METHODS,
			   Cap#wsp_capabilities.extended_methods,
			   Def#wsp_capabilities.extended_methods),
	 encode_capability(?WSP_CAP_HEADER_CODE_PAGES,
			   Cap#wsp_capabilities.header_code_pages,
			   Def#wsp_capabilities.header_code_pages),
	 encode_capability(?WSP_CAP_CLIENT_MESSAGE_SIZE,
			   Cap#wsp_capabilities.client_message_size,
			   Def#wsp_capabilities.client_message_size),
	 encode_capability(?WSP_CAP_SERVER_MESSAGE_SIZE,
			   Cap#wsp_capabilities.server_message_size,
			   Def#wsp_capabilities.server_message_size)],
    Unknown =
	lists:map(fun({Id, Data}) when integer(Id) ->
			  <<1:1, Id:7, Data/binary>>;
		     ({Id,Data}) ->
			  <<(encode_text_string(Id))/binary, Data/binary>>
			      end, Cap#wsp_capabilities.unknown),
    list_to_binary(
      lists:map(fun(<<>>) -> [];
		   (Bin) ->
			[e_uintvar(size(Bin)), Bin]
		end, Known ++ Unknown)).




encode_capability(_Capa, Default, Default) ->
    <<>>;
encode_capability(Capa, Value, _) ->
    case Capa of
	?WSP_CAP_ALIASES ->
	    <<1:1, ?WSP_CAP_ALIASES:7, (encode_addresses(Value))/binary>>;

	?WSP_CAP_CLIENT_SDU_SIZE ->
	    <<1:1, ?WSP_CAP_CLIENT_SDU_SIZE:7, (e_uintvar(Value))/binary>>;

	?WSP_CAP_SERVER_SDU_SIZE ->
	    <<1:1, ?WSP_CAP_SERVER_SDU_SIZE:7, (e_uintvar(Value))/binary>>;

	?WSP_CAP_PROTOCOL_OPTIONS ->
	    Opts = case lists:member(confirmed_push, Value) of
		       true -> 16#80;
		       false -> 0
		   end bor
		   case lists:member(push, Value) of
		       true -> 16#40;
		       false -> 0
		   end bor
		   case lists:member(resume, Value) of
		       true -> 16#20;
		       false -> 0
		   end bor
		   case lists:member(acknowledgement_headers, Value) of
		       true -> 16#10;
		       false -> 0
		   end,
	    %% FIXME: symbolic encode/decode of options
	    <<1:1, ?WSP_CAP_PROTOCOL_OPTIONS:7, Opts>>;

	?WSP_CAP_METHOD_MOR ->
	    <<1:1, ?WSP_CAP_METHOD_MOR:7, (e_uintvar(Value))/binary>>;

	?WSP_CAP_PUSH_MOR ->
	    <<1:1, ?WSP_CAP_PUSH_MOR:7, (e_uintvar(Value))/binary>>;

	?WSP_CAP_EXTENDED_METHODS ->
	    <<1:1, ?WSP_CAP_EXTENDED_METHODS:7,
	     (encode_extended_methods(Value))/binary>>;

	?WSP_CAP_HEADER_CODE_PAGES ->
	    Data = list_to_binary(
		     lists:map(fun(Page) when integer(Page) -> Page;
				  ({Page,Name}) ->
				       [Page, encode_text_string(Name)]
			       end, Value)),
	    <<1:1, ?WSP_CAP_HEADER_CODE_PAGES:7, Data/binary>>;

	?WSP_CAP_CLIENT_MESSAGE_SIZE ->
	    <<1:1, ?WSP_CAP_CLIENT_MESSAGE_SIZE:7,
	     (e_uintvar(Value))/binary>>;

	?WSP_CAP_SERVER_MESSAGE_SIZE ->
	    <<1:1, ?WSP_CAP_SERVER_MESSAGE_SIZE:7,
	     (e_uintvar(Value))/binary>>;
	_ when integer(Capa) ->
	    <<1:1, Capa:7, Value/binary>>;
	_ when list(Capa) ->
	    <<(encode_text_string(Capa))/binary, Value/binary>>
   end.


decode_capabilities(<<>>, WspCaps) ->
    WspCaps;
decode_capabilities(D0,WspCaps) ->
    {Len, D1} = d_uintvar(D0),
    <<Capa:Len/binary, D2/binary>> = D1,
    WspCaps1 =
	case Capa of
	    <<1:1, Id:7, Data/binary>> ->
		decode_capa(Id, Data, WspCaps);
	    _ ->
		{Id,Data} = d_text_string(Capa),
		decode_capa(Id, Data, WspCaps)
	end,
    decode_capabilities(D2, WspCaps1).



decode_capa(Id,Data, WspCaps) ->
    case Id of
	?WSP_CAP_SERVER_SDU_SIZE ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{server_sdu_size=Val};

	?WSP_CAP_CLIENT_SDU_SIZE ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{client_sdu_size=Val};

	?WSP_CAP_PROTOCOL_OPTIONS ->
	    <<POP,_/binary>> = Data,
	    Opts =
		if POP band 16#80 == 16#80 -> [confirmed_push];
		   true -> []
		end ++
		if POP band 16#40 == 16#40 -> [push];
		   true -> []
		end ++
		if POP band 16#20 == 16#20 -> [resume];
		   true -> []
		end ++
		if POP band 16#10 == 16#10 -> [acknowledgement_headers];
		   true -> []
		end,
	    WspCaps#wsp_capabilities{protocol_options=Opts};

	?WSP_CAP_METHOD_MOR ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{method_mor=Val};

	?WSP_CAP_PUSH_MOR ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{push_mor=Val};

	?WSP_CAP_EXTENDED_METHODS ->
	    Extended = decode_extended_methods(Data),
	    WspCaps#wsp_capabilities { extended_methods = Extended };

	?WSP_CAP_HEADER_CODE_PAGES ->
	    %% Client send [Code(uint8) Name(text-string)]*
	    %% Server send [Code(uint8)]*
	    io:format("FIXME: Header Code Pages = ~p\n",[Data]),
	    WspCaps;

	?WSP_CAP_ALIASES ->
	    Aliases = decode_addresses(Data),
	    WspCaps#wsp_capabilities { aliases = Aliases };

	?WSP_CAP_CLIENT_MESSAGE_SIZE ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{client_message_size=Val};

	?WSP_CAP_SERVER_MESSAGE_SIZE ->
	    {Val,_} = d_uintvar(Data),
	    WspCaps#wsp_capabilities{server_message_size=Val};
	_  ->
	    Unknown = [{Id, Data} | WspCaps#wsp_capabilities.unknown],
	    io:format("WARNING: ignoring unknown capability ~p\n",
		      [Unknown]),
	    WspCaps#wsp_capabilities{unknown = Unknown}
    end.


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Headers = [ Header ]
%% Header  = {FieldName, FieldValue}
%% FieldName = atom()
%% FieldValue = {Value, Params}
%%            | Value
%%
%% Params = [{Param,Value} | Param]
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

-define(WH(Name,Value,Params),
	#wsp_header { name = (Name), value = (Value), params = Params}).

encode_headers(Headers) ->
    encode_headers(Headers, ?WSP_DEFAULT_VERSION).

encode_headers(Headers, Version) ->
    encode_headers(Headers, Version, []).

encode_headers([H|T], Version, Acc) ->
    encode_headers(T, Version, [encode_header(H, Version)|Acc]);
encode_headers([], _,  Acc) ->
    list_to_binary(lists:reverse(Acc)).


decode_headers(Bin) ->
    decode_headers(Bin, ?WSP_DEFAULT_VERSION).

decode_headers(<<>>, _Version) ->
    [];
decode_headers(Data, Version) ->
    decode_headers(Data, [], Version, ?WSP_DEFAULT_CODEPAGE).


decode_headers(<<1:1,Code:7,Data/binary>>,Acc,Version,CP) ->
    FieldName = lookup_field_name(Code),
    {FieldData,Data1} = scan_header_data(Data),
    H = decode_header(FieldName, FieldData,Version,CP),
    ?dbg("header: ~p, field data=~p, header=~p\n",
	 [FieldName, FieldData, H]),
    if H#wsp_header.name == 'Encoding-Version' ->
	    Version1 = H#wsp_header.value,
	    ?dbg("Version switch from ~w to ~w\n", [Version, Version1]),
	    decode_headers(Data1,[H|Acc],Version1, CP);
       true ->
	    decode_headers(Data1,[H|Acc],Version, CP)
    end;
decode_headers(Data = <<Code,_/binary>>,Acc,Version,CP)
  when Code >= 32, Code < 127->
    {TmpField,Data1} = d_text_string(Data),
    FieldName = normalise_field_name(TmpField),
    {FieldData,Data2} = scan_header_data(Data1),
    H = decode_header(FieldName,FieldData,Version,CP),
    ?dbg("header: ~p, field data=~p, header=~p\n",
	 [FieldName, FieldData, H]),
    if H#wsp_header.name == 'Encoding-Version' ->
	    Version1 = H#wsp_header.value,
	    ?dbg("Version switch from ~w to ~w\n", [Version, Version1]),
	    decode_headers(Data2,[H|Acc],Version1, CP);
       true ->
	    decode_headers(Data2,[H|Acc],Version, CP)
    end;
decode_headers(<<CP1,Data/binary>>,Acc,Version,_CP) when CP1 >= 1, CP1 =< 31 ->
    ?dbg("decode_headers: codpage changed form ~w -> ~w\n",[_CP,CP1]),
    decode_headers(Data,Acc,Version,CP1);
decode_headers(<<16#7f,CP1,Data/binary>>,Acc,Version,_CP) ->
    ?dbg("decode_headers: codpage changed form ~w -> ~w\n",[_CP,CP1]),
    decode_headers(Data,Acc,Version,CP1);

decode_headers(<<>>, Acc, _Version, _CP) ->
    lists:reverse(Acc).

%%
%% Retrive the header data
%%  (this makes it possible to skip unknown encoding)
%%
scan_header_data(Data = <<N,Data0/binary>>) ->
    if N >= 0, N =< 30 ->
	    <<Value:N/binary, Data1/binary>> = Data0,
	    {{short,Value}, Data1};
       N == 31 ->
	    {N1, Data1} = d_uintvar(Data0),
	    <<Value:N1/binary, Data2/binary>> = Data1,
	    {{long,Value}, Data2};
       N >= 32, N =< 127 ->
	    d_text_string(Data);
       true ->
	    { N band 16#7f, Data0}
    end.

%%
%% Decode header: return #wsp_header
%%
decode_header(Field, Value) ->
    decode_header(Field, Value,
		  ?WSP_DEFAULT_VERSION,
		  ?WSP_DEFAULT_CODEPAGE).

decode_header(Field, Value, Version, 1) ->
    case Field of
	'Accept' ->
	    decode_accept(Value, Version);

	'Accept-Charset' when Version >= ?WSP_13 ->
	    decode_accept_charset(Value, Version);
	'Accept-Charset' ->
	    decode_accept_charset(Value, Version);

	'Accept-Encoding' when Version >= ?WSP_13 ->
	    decode_accept_encoding(Value, Version);
	'Accept-Encoding' ->
	    decode_accept_encoding(Value, Version);

	'Accept-Language' ->
	    decode_accept_language(Value, Version);
	'Accept-Ranges' ->
	    decode_accept_ranges(Value, Version);
	'Age' ->
	    decode_age(Value,Version);
	'Allow' ->
	    decode_allow(Value,Version);
	'Authorization' ->
	    decode_authorization(Value,Version);

	'Cache-Control' when Version >= ?WSP_14 ->
	    decode_cache_control(Value,Version);
	'Cache-Control' when Version >= ?WSP_13 ->
	    decode_cache_control(Value,Version);
	'Cache-Control' ->
	    decode_cache_control(Value,Version);

	'Connection' ->
	    decode_connection(Value,Version);
	'Content-Base' ->
	    decode_content_base(Value,Version);
	'Content-Encoding' ->
	    decode_content_encoding(Value,Version);
	'Content-Language' ->
	    decode_content_language(Value,Version);
	'Content-Length' ->
	    decode_content_length(Value,Version);
	'Content-Location' ->
	    decode_content_location(Value,Version);
	'Content-Md5' ->
	    decode_content_md5(Value,Version);

	'Content-Range' when Version >= ?WSP_13 ->
	    decode_content_range(Value,Version);
	'Content-Range' ->
	    decode_content_range(Value,Version);

	'Content-Type' ->
	    decode_content_type(Value,Version);
	'Date' ->
	    decode_date(Value, Version);
	'Etag' ->
	    decode_etag(Value,Version);
	'Expires' ->
	    decode_expires(Value,Version);
	'From' ->
	    decode_from(Value,Version);
	'Host' ->
	    decode_host(Value,Version);
	'If-Modified-Since' ->
	    decode_if_modified_since(Value,Version);
	'If-Match' ->
	    decode_if_match(Value,Version);
	'If-None-Match' ->
	    decode_if_none_match(Value,Version);
	'If-Range' ->
	    decode_if_range(Value,Version);
	'If-Unmodified-Since' ->
	    decode_if_unmodified_since(Value,Version);
	'Location' ->
	    decode_location(Value,Version);
	'Last-Modified' ->
	    decode_last_modified(Value,Version);
	'Max-Forwards' ->
	    decode_max_forwards(Value,Version);
	'Pragma' ->
	    decode_pragma(Value,Version);
	'Proxy-Authenticate' ->
	    decode_proxy_authenticate(Value,Version);
	'Proxy-Authorization' ->
	    decode_proxy_authorization(Value,Version);
	'Public' ->
	    decode_public(Value,Version);
	'Range' ->
	    decode_range(Value,Version);
	'Referer' ->
	    decode_referer(Value,Version);
	'Retry-After' ->
	    decode_retry_after(Value,Version);
	'Server' ->
	    decode_server(Value,Version);
	'Transfer-Encoding' ->
	    decode_transfer_encoding(Value,Version);
	'Upgrade' ->
	    decode_upgrade(Value,Version);
	'User-Agent' ->
	    decode_user_agent(Value,Version);
	'Vary' ->
	    decode_vary(Value,Version);
	'Via' ->
	    decode_via(Value,Version);
	'Warning' ->
	    decode_warning(Value,Version);
	'Www-Authenticate' ->
	    decode_www_authenticate(Value,Version);

	'Content-Disposition' when Version >= ?WSP_14 ->
	    decode_content_disposition(Value,Version);
	'Content-Disposition' ->
	    decode_content_disposition(Value,Version);

	'X-Wap-Application-Id' when Version >= ?WSP_12 ->
	    decode_x_wap_application_id(Value,Version);

	'X-Wap-Content-Uri' when Version >= ?WSP_12 ->
	    decode_x_wap_content_uri(Value,Version);

	'X-Wap-Initiator-Uri' when Version >= ?WSP_12 ->
	    decode_x_wap_initiator_uri(Value,Version);

	'Accept-Application' when Version >= ?WSP_12 ->
	    decode_accept_application(Value,Version);

	'Bearer-Indication' when Version >= ?WSP_12 ->
	    decode_bearer_indication(Value,Version);

	'Push-Flag' when Version >= ?WSP_12 ->
	    decode_push_flag(Value,Version);

	'Profile' when Version >= ?WSP_12 ->
	    decode_profile(Value,Version);

	'Profile-Diff' when Version >= ?WSP_12 ->
	    decode_profile_diff(Value,Version);

	'Profile-Warning' when Version >= ?WSP_12 ->
	    decode_profile_warning(Value,Version);

	'Expect' when Version >= ?WSP_15 ->
	    decode_expect(Value,Version);
	'Expect' when Version >= ?WSP_13 ->
	    decode_expect(Value,Version);

	'Te' when Version >= ?WSP_13 ->
	    decode_te(Value,Version);
	'Trailer' when Version >= ?WSP_13 ->
	    decode_trailer(Value,Version);

	'X-Wap-Tod' when Version >= ?WSP_13 ->
	    decode_x_wap_tod(Value,Version);
	'X-Wap.tod' when Version >= ?WSP_13 ->
	    decode_x_wap_tod(Value,Version);

	'Content-Id' when Version >= ?WSP_13 ->
	    decode_content_id(Value,Version);
	'Set-Cookie' when Version >= ?WSP_13 ->
	    decode_set_cookie(Value,Version);
	'Cookie' when Version >= ?WSP_13 ->
	    decode_cookie(Value,Version);

	'Encoding-Version' when Version >= ?WSP_13 ->
	    decode_encoding_version(Value,Version);
	'Profile-Warning' when Version >= ?WSP_14 ->
	    decode_profile_warning(Value,Version);

	'X-Wap-Security' when Version >= ?WSP_14 ->
	    decode_x_wap_security(Value,Version);
	'X-Wap-Loc-Invocation' when Version >= ?WSP_15 ->
	    decode_x_wap_loc_invocation(Value,Version); %% ???
	'X-Wap-Loc-Delivery' when Version >= ?WSP_15 ->
	    decode_x_wap_loc_delivery(Value,Version);   %% ???
	_ ->
	    ?dbg("Warning: none standard field ~p in version ~p codepage=1\n",
		 [Field, Version]),
	    ?WH(Field, Value, [])
    end;
decode_header(Field, Value, _Version, _CP) ->
    ?dbg("Warning: none standard field ~p in version ~p codepage=~w\n",
	 [Field, _Version, _CP]),
    ?WH(Field, Value, []).

%%
%% Encode field and value according to version
%% FIXME: spilt multiple header values (i.e Via) into multiple
%%        headers
%%
encode_header(H, Version) ->
    case H#wsp_header.name of
	'Accept' ->
	    [16#80, encode_accept(H, Version)];
	'Accept-Charset' when Version >= ?WSP_13 ->
	    [16#bb, encode_accept_charset(H, Version)];
	'Accept-Charset' ->
	    [16#81, encode_accept_charset(H, Version)];
	'Accept-Encoding' when Version >= ?WSP_13 ->
	    [16#bc, encode_accept_encoding(H, Version)];
	'Accept-Encoding' ->
	    [16#82, encode_accept_encoding(H, Version)];
	'Accept-Language' ->
	    [16#83, encode_accept_language(H, Version)];
	'Accept-Ranges' ->
	    [16#84, encode_accept_ranges(H, Version)];
	'Accept-Application' when Version >= ?WSP_12 ->
	    [16#b2, encode_accept_application(H,Version)];
	'Age' ->
	    [16#85, encode_age(H, Version)];
	'Allow' ->
	    [16#86, encode_allow(H, Version)];
	'Authorization' ->
	    [16#87, encode_authorization(H, Version)];
	'Cache-Control' when Version >= ?WSP_14 ->
	    [16#c7, encode_cache_control(H, Version)];
	'Cache-Control' when Version >= ?WSP_13 ->
	    [16#bd, encode_cache_control(H, Version)];
	'Cache-Control' ->
	    [16#88, encode_cache_control(H, Version)];
	'Connection' ->
	    [16#89, encode_connection(H, Version)];
	'Content-Base' ->
	    [16#8a, encode_content_base(H, Version)];
	'Content-Encoding' ->
	    [16#8b, encode_content_encoding(H, Version)];

	'Content-Language' ->
	    [16#8c, encode_content_language(H,Version)];
	'Content-Length' ->
	    [16#8d, encode_content_length(H,Version)];
	'Content-Location' ->
	    [16#8e, encode_content_location(H,Version)];
	'Content-Md5' ->
	    [16#8f, encode_content_md5(H,Version)];
	'Content-Range' when Version >= ?WSP_13 ->
	    [16#be, encode_content_range(H,Version)];
	'Content-Range' ->
	    [16#90, encode_content_range(H,Version)];
	'Content-Type' ->
	    [16#91, encode_content_type(H,Version)];
	'Date' ->
	    [16#92, encode_date(H,Version)];
	'Etag' ->
	    [16#93, encode_etag(H,Version)];
	'Expires' ->
	    [16#94, encode_expires(H,Version)];
	'From' ->
	    [16#95, encode_from(H,Version)];
	'Host' ->
	    [16#96, encode_host(H,Version)];
	'If-Modified-Since' ->
	    [16#97, encode_if_modified_since(H,Version)];
	'If-Match' ->
	    [16#98, encode_if_match(H,Version)];
	'If-None-Match' ->
	    [16#99, encode_if_none_match(H,Version)];
	'If-Range' ->
	    [16#9a, encode_if_range(H,Version)];
	'If-Unmodified-Since' ->
	    [16#9b, encode_if_unmodified_since(H,Version)];
	'Location' ->
	    [16#9c, encode_location(H,Version)];
	'Last-Modified' ->
	    [16#9d, encode_last_modified(H,Version)];
	'Max-Forwards' ->
	    [16#9e, encode_max_forwards(H,Version)];
	'Pragma' ->
	    [16#9f, encode_pragma(H,Version)];
	'Proxy-Authenticate' ->
	    [16#a0, encode_proxy_authenticate(H,Version)];
	'Proxy-Authorization' ->
	    [16#a1, encode_proxy_authorization(H,Version)];
	'Public' ->
	    [16#a2, encode_public(H,Version)];
	'Range' ->
	    [16#a3, encode_range(H,Version)];
	'Referer' ->
	    [16#a4, encode_referer(H,Version)];
	'Retry-After' ->
	    [16#a5, encode_retry_after(H,Version)];
	'Server' ->
	    [16#a6, encode_server(H,Version)];
	'Transfer-Encoding' ->
	    [16#a7, encode_transfer_encoding(H,Version)];
	'Upgrade' ->
	    [16#a8, encode_upgrade(H,Version)];
	'User-Agent' ->
	    [16#a9, encode_user_agent(H,Version)];
	'Vary' ->
	    [16#aa, encode_vary(H,Version)];
	'Via' ->
	    [16#ab, encode_via(H,Version)];
	'Warning' ->
	    [16#ac, encode_warning(H,Version)];
	'Www-Authenticate' ->
	    [16#ad, encode_www_authenticate(H,Version)];

	'Content-Disposition' when Version >= ?WSP_14 ->
	    [16#c5, encode_content_disposition(H,Version)];
	'Content-Disposition' ->
	    [16#ae, encode_content_disposition(H,Version)];


	'X-Wap-Application-Id' when Version >= ?WSP_12 ->
	    [16#af, encode_x_wap_application_id(H,Version)];
	'X-Wap-Content-Uri' when Version >= ?WSP_12 ->
	    [16#b0, encode_x_wap_content_uri(H,Version)];
	'X-Wap-Initiator-Uri' when Version >= ?WSP_12 ->
	    [16#b1, encode_x_wap_initiator_uri(H,Version)];

	'Bearer-Indication' when Version >= ?WSP_12 ->
	    [16#b3, encode_bearer_indication(H,Version)];
	'Push-Flag' when Version >= ?WSP_12 ->
	    [16#b4, encode_push_flag(H,Version)];

	'Profile' when Version >= ?WSP_12 ->
	    [16#b5, encode_profile(H,Version)];
	'Profile-Diff' when Version >= ?WSP_12 ->
	    [16#b6, encode_profile_diff(H,Version)];
	'Profile-Warning' when Version >= ?WSP_14 ->
	    [16#c4, encode_profile_warning(H,Version)];
	'Profile-Warning' when Version >= ?WSP_12 ->
	    [16#b7, encode_profile_warning(H,Version)];

	'Expect' when Version >= ?WSP_15 ->
	    [16#c8, encode_expect(H,Version)];
	'Expect' when Version >= ?WSP_13 ->
	    [16#b8, encode_expect(H,Version)];
	'Te' when Version >= ?WSP_13 ->
	    [16#b9, encode_te(H,Version)];
	'Trailer' when Version >= ?WSP_13 ->
	    [16#ba, encode_trailer(H,Version)];
	'X-Wap-Tod' when Version >= ?WSP_13 ->
	    [16#bf, encode_x_wap_tod(H,Version)];
	'Content-Id' when Version >= ?WSP_13 ->
	    [16#c0, encode_content_id(H,Version)];
	'Set-Cookie' when Version >= ?WSP_13 ->
	    [16#c1, encode_set_cookie(H,Version)];
	'Cookie' when Version >= ?WSP_13 ->
	    [16#c2, encode_cookie(H,Version)];
	'Encoding-Version' when Version >= ?WSP_13 ->
	    [16#c3, encode_encoding_version(H,Version)];
	'Encoding-Version' when Version < ?WSP_13 ->
	    [encode_text_string("Encoding-Version"),
	     encode_text_string(lists:flatten(format_version(H#wsp_header.value)))];

	'X-Wap-Security' when Version >= ?WSP_14 ->
	    [16#c6, encode_x_wap_security(H,Version)];
	'X-Wap-Loc-Invocation' when Version >= ?WSP_15 ->
	    [16#c9, encode_x_wap_loc_invocation(H,Version)];
	'X-Wap-Loc-Delivery' when Version >= ?WSP_15 ->
	    [16#ca, encode_x_wap_loc_delivery(H,Version)];
	Field when atom(Field) ->
	    [encode_text_string(atom_to_list(Field)),
	     encode_text_string(H#wsp_header.value)];
	Field when list(Field) ->
	    [encode_text_string(Field),
	     encode_text_string(H#wsp_header.value)]
    end.

%%
%% Convert HTTP headers into WSP headers
%%
parse_headers([H | Hs]) ->
    parse_header(H, Hs);
parse_headers([]) ->
    [].

parse_header(H) ->
    parse_header(H, []).

parse_header({FieldName,FieldValue}, Hs) ->
    case single_comma_field(FieldName) of
	true ->
	    io:format("parse: ~s: ~s\n", [FieldName, FieldValue]),
	    H = parse_hdr(FieldName,FieldValue),
	    io:format("header: ~p\n", [H]),
	    [H | parse_headers(Hs)];
	false ->
	    Values = string:tokens(FieldValue, ","),
	    parse_header(FieldName, Values, Hs)
    end.

parse_header(FieldName, [Value|Vs], Hs) ->
    io:format("parse: ~s: ~s\n", [FieldName, Value]),
    H = parse_hdr(FieldName, Value),
    io:format("header: ~p\n", [H]),
    [H | parse_header(FieldName, Vs, Hs)];
parse_header(_FieldName, [], Hs) ->
    parse_headers(Hs).


single_comma_field(Field) ->
    case Field of
	'Set-Cookie' -> true; %% FIXME (Is multiple!)
	'Date' -> true;
	'Expires' -> true;
	'If-Modified-Since' -> true;
	'If-Range' -> true;
	'If-Unmodified-Since' -> true;
	'Last-Modified' -> true;
	'Retry-After' -> true;
	'X-Wap-Tod' -> true;
	_ -> false
    end.


parse_hdr(Field, Value0) ->
    Value = trim(Value0),
    case Field of
	'Accept' ->           parse_accept(Value);
	'Accept-Charset' ->   parse_accept_charset(Value);
	'Accept-Encoding' ->  parse_accept_encoding(Value);
	'Accept-Language' ->  parse_accept_language(Value);
	'Accept-Ranges' ->    parse_accept_ranges(Value);
	'Age' ->              parse_age(Value);
	'Allow' ->            parse_allow(Value);
	'Authorization' ->    parse_authorization(Value);
	'Cache-Control' ->    parse_cache_control(Value);
	'Connection' ->       parse_connection(Value);
	'Content-Base' ->     parse_content_base(Value);
	'Content-Encoding' -> parse_content_encoding(Value);
	'Content-Language' -> parse_content_language(Value);
	'Content-Length' ->   parse_content_length(Value);
	'Content-Location' -> parse_content_location(Value);
	'Content-Md5' ->      parse_content_md5(Value);
	'Content-Range' ->    parse_content_range(Value);
	'Content-Type' ->     parse_content_type(Value);
	'Date' ->             parse_date(Value);
	'Etag' ->             parse_etag(Value);
	'Expires' ->          parse_expires(Value);
	'From' ->             parse_from(Value);
	'Host' ->             parse_host(Value);
	'If-Modified-Since' -> parse_if_modified_since(Value);
	'If-Match' ->         parse_if_match(Value);
	'If-None-Match' ->    parse_if_none_match(Value);
	'If-Range' ->         parse_if_range(Value);
	'If-Unmodified-Since' -> parse_if_unmodified_since(Value);
	'Location' ->         parse_location(Value);
	'Last-Modified' ->    parse_last_modified(Value);
	'Max-Forwards' ->     parse_max_forwards(Value);
	'Pragma' ->           parse_pragma(Value);
	'Proxy-Authenticate' -> parse_proxy_authenticate(Value);
	'Proxy-Authorization' -> parse_proxy_authorization(Value);
	'Public' ->           parse_public(Value);
	'Range' ->            parse_range(Value);
	'Referer' ->          parse_referer(Value);
	'Retry-After' ->      parse_retry_after(Value);
	'Server' ->           parse_server(Value);
	'Transfer-Encoding' -> parse_transfer_encoding(Value);
	'Upgrade' ->          parse_upgrade(Value);
	'User-Agent' ->       parse_user_agent(Value);
	'Vary' ->             parse_vary(Value);
	'Via' ->              parse_via(Value);
	'Warning' ->          parse_warning(Value);
	'Www-Authenticate' -> parse_www_authenticate(Value);
	'Content-Disposition' -> parse_content_disposition(Value);
	'X-Wap-Application-Id' -> parse_x_wap_application_id(Value);
	'X-Wap-Content-Uri' -> parse_x_wap_content_uri(Value);
	'X-Wap-Initiator-Uri' -> parse_x_wap_initiator_uri(Value);
	'Accept-Application' -> parse_accept_application(Value);
	'Bearer-Indication' -> parse_bearer_indication(Value);
	'Push-Flag' ->        parse_push_flag(Value);
	'Profile' ->          parse_profile(Value);
	'Profile-Diff' ->     parse_profile_diff(Value);
	'Profile-Warning' ->  parse_profile_warning(Value);
	'Expect' ->           parse_expect(Value);
	'Te' ->               parse_te(Value);
	'Trailer' ->          parse_trailer(Value);
	'X-Wap-Tod' ->        parse_x_wap_tod(Value);
	'Content-Id' ->       parse_content_id(Value);
	'Set-Cookie' ->       parse_set_cookie(Value);
	'Cookie' ->           parse_cookie(Value);
	'Encoding-Version' -> parse_encoding_version(Value);
	'X-Wap-Security' ->   parse_x_wap_security(Value);
	'X-Wap-Loc-Invocation' -> parse_x_wap_loc_invocation(Value);
	'X-Wap-Loc-Delivery' -> parse_x_wap_loc_delivery(Value);
	_ ->
	    ?dbg("Warning: header field ~p not recognissed\n",[Field]),
	    #wsp_header { name = Field, value = Value}
    end.

%%
%% Format headers, will combine multiple headers into one
%% FIXME: if length is < MAX_HTTP_HEADER_LENGTH
%%
format_headers(Hs) ->
    format_hdrs(lists:keysort(#wsp_header.name,Hs), []).

format_hdrs([H | Hs], Acc) ->
    V1 = format_value(H),
    format_hdrs(Hs, H#wsp_header.name, V1, Acc);
format_hdrs([], Acc) ->
    lists:reverse(Acc).

format_hdrs([H|Hs], FieldName, FieldValue, Acc)
  when FieldName == H#wsp_header.name ->
    V1 = format_value(H),
    format_hdrs(Hs, FieldName, [FieldValue,",",V1], Acc);
format_hdrs(Hs, FieldName, FieldValue, Acc) ->
    format_hdrs(Hs, [{FieldName, lists:flatten(FieldValue)} | Acc]).


%%
%% Format header: #wsp_header => {FieldName, Value}
%%

format_header(H) ->
    {H#wsp_header.name, format_value(H)}.

format_value(H) ->
    case H#wsp_header.name of
	'Accept' ->   format_accept(H);
	'Accept-Charset' ->   format_accept_charset(H);
	'Accept-Encoding' ->  format_accept_encoding(H);
	'Accept-Language' ->  format_accept_language(H);
	'Accept-Ranges' ->    format_accept_ranges(H);
	'Age' ->     format_age(H);
	'Allow' ->   format_allow(H);
	'Authorization' ->   format_authorization(H);
	'Cache-Control' ->   format_cache_control(H);
	'Connection' ->      format_connection(H);
	'Content-Base' ->    format_content_base(H);
	'Content-Encoding' -> format_content_encoding(H);
	'Content-Language' -> format_content_language(H);
	'Content-Length' -> format_content_length(H);
	'Content-Location' -> format_content_location(H);
	'Content-Md5' -> format_content_md5(H);
	'Content-Range' -> format_content_range(H);
	'Content-Type' -> format_content_type(H);
	'Date' -> format_date(H);
	'Etag' -> format_etag(H);
	'Expires' -> format_expires(H);
	'From' -> format_from(H);
	'Host' -> format_host(H);
	'If-Modified-Since' -> format_if_modified_since(H);
	'If-Match' -> format_if_match(H);
	'If-None-Match' -> format_if_none_match(H);
	'If-Range' -> format_if_range(H);
	'If-Unmodified-Since' -> format_if_unmodified_since(H);
	'Location' -> format_location(H);
	'Last-Modified' -> format_last_modified(H);
	'Max-Forwards' -> format_max_forwards(H);
	'Pragma' -> format_pragma(H);
	'Proxy-Authenticate' -> format_proxy_authenticate(H);
	'Proxy-Authorization' -> format_proxy_authorization(H);
	'Public' -> format_public(H);
	'Range' -> format_range(H);
	'Referer' -> format_referer(H);
	'Retry-After' -> format_retry_after(H);
	'Server' -> format_server(H);
	'Transfer-Encoding' -> format_transfer_encoding(H);
	'Upgrade' -> format_upgrade(H);
	'User-Agent' -> format_user_agent(H);
	'Vary' -> format_vary(H);
	'Via' -> format_via(H);
	'Warning' -> format_warning(H);
	'Www-Authenticate' -> format_www_authenticate(H);
	'Content-Disposition' -> format_content_disposition(H);
	'X-Wap-Application-Id' -> format_x_wap_application_id(H);
	'X-Wap-Content-Uri' -> format_x_wap_content_uri(H);
	'X-Wap-Initiator-Uri' -> format_x_wap_initiator_uri(H);
	'Accept-Application' -> format_accept_application(H);
	'Bearer-Indication' -> format_bearer_indication(H);
	'Push-Flag' -> format_push_flag(H);
	'Profile' -> format_profile(H);
	'Profile-Diff' -> format_profile_diff(H);
	'Profile-Warning' -> format_profile_warning(H);
	'Expect' ->  format_expect(H);
	'Te' -> format_te(H);
	'Trailer' -> format_trailer(H);
	'X-Wap-Tod' ->  format_x_wap_tod(H);
	'Content-Id' -> format_content_id(H);
	'Set-Cookie' -> format_set_cookie(H);
	'Cookie' -> format_cookie(H);
	'Encoding-Version' -> format_encoding_version(H);
	'X-Wap-Security' -> format_x_wap_security(H);
	'X-Wap-Loc-Invocation' -> format_x_wap_loc_invocation(H);
	'X-Wap-Loc-Delivery' -> format_x_wap_loc_delivery(H);
	_Field ->
	    ?dbg("Warning: header field ~s not recognissed\n",[_Field]),
	    to_list(H#wsp_header.value)
    end.


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Encode of field values
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Accept: <content-type> [q=<q-value>] [params]
%% Type: Multiple
%% Ref: 8.4.2.7
%%
%%  Accept-value = Constrained-media | Accept-general-form
%%
%%  Accept-general-form = Value-length Media-range [Accept-parameters]
%%  Media-range = (Well-known-media | Extension-media) *(Parameter)
%%  Accept-parameters = Q-token Q-value *(Accept-extension)
%%  Accept-extension = Parameter
%%  Constrain-media = Constrained-encoding
%%  Well-known-media = Integer-value
%%  Constrained-encoding = Short-Integer | Extension-media
%%  Q-token = <Octet 128>
%%
parse_accept(String) ->
    %% FIXME
    ?WH('Accept',String,[]).

format_accept(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_accept(H, Version) ->
    case encode_params(H#wsp_header.params,Version) of
	<<>> ->
	    encode_well_known_media(H#wsp_header.value, Version);
	Params ->
	    Media = encode_well_known_media(H#wsp_header.value, Version),
	    e_value(Media, Params)
    end.

decode_accept(Value, Version) when integer(Value) ->
    %% Constrained-encoding: Short-Integer
    ?WH('Accept',decode_well_known_media(Value, Version),[]);
decode_accept(Value, Version) when list(Value) ->
    ?WH('Accept',decode_well_known_media(Value,Version),[]);
decode_accept({_,Data}, Version)  ->
    %% Accept-general-form
    {Value,QData} = scan_header_data(Data),
    Media_Range = decode_well_known_media(Value,Version),
    Params =  decode_params(QData, Version),
    ?WH('Accept',Media_Range,Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Accept-Charset: <charset> | * [q=<q-value>]
%% Type: Multiple
%% Ref: 8.4.2.8
%% Note that the definition of this one is a mess!!!!
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_accept_charset(String) ->
    %% FIXME
    ?WH('Accept-Charset',String,[]).

format_accept_charset(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_accept_charset(H, _Version) ->
    %% FIXME
    encode_text_string(H#wsp_header.value).

decode_accept_charset(0, _Version) ->
    ?WH('Accept-Charset',"*",[]);
decode_accept_charset(Value, _Version) when integer(Value) ->
    ?WH('Accept-Charset', decode_charset(Value),[]);
decode_accept_charset(Value, _Version) when list(Value) ->
    ?WH('Accept-Charset',Value,[]);
decode_accept_charset({short,Data}, _Version) ->
    %% Me guessing that the short form SHOULD be mulit octet integer!!!
    Value = d_long(Data),
    ?WH('Accept-Charset', decode_charset(Value),[]);
decode_accept_charset({long,Value}, _Version) ->
    {Data1, QData} = scan_header_data(Value),
    CharSet = case Data1 of
		  0 ->
		      "*";
		  Value1 when integer(Value1) ->
		      decode_charset(Value1);
		  Value1 when list(Value1) ->
		      Value1;
		  {short,Value1} ->
		      Value2 = d_long(Value1),
		      decode_charset(Value2)
	      end,
    Params = if QData == <<>> ->
		     [];
		true ->
		     {QValue,_} = d_q_value(QData),
		     {CharSet,[{q, QValue}]}
	     end,
    ?WH('Accept-Charset',CharSet, Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Accept-Encoding: gzip | compress | deflate | * [q=<q-value>]
%% Ref:
%% Type: Multiple
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_accept_encoding(String) ->
    ?WH('Accept-Encoding',String,[]).

format_accept_encoding(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_accept_encoding(H, _Version) ->
    %% FIXME general form
    case H#wsp_header.value of
	"gzip"     -> ?ENCODE_SHORT(0);
	"compress" -> ?ENCODE_SHORT(1);
	"deflate"  -> ?ENCODE_SHORT(2);
	Value      -> encode_text_string(Value)
    end.

decode_accept_encoding(0, _Version) ->
    ?WH('Accept-Encoding',"gzip",[]);
decode_accept_encoding(1, _Version) ->
    ?WH('Accept-Encoding',"compress",[]);
decode_accept_encoding(2, _Version) ->
    ?WH('Accept-Encoding',"deflate",[]);
decode_accept_encoding(Value, Version) when list(Version) ->
    ?WH('Accept-Encoding',Value,[]);
decode_accept_encoding({_,Data}, _Version) when binary(Data) ->
    {Enc, Data1} = scan_header_data(Data),
    Params = if Data1 == <<>> ->
		     [];
		true ->
		     {QVal,_} = d_q_value(Data1),
		     [{q, QVal}]
	     end,
    case Enc of
	0 -> ?WH('Accept-Encoding',"gzip",Params);
	1 -> ?WH('Accept-Encoding',"compress",Params);
	2 -> ?WH('Accept-Encoding',"deflate",Params);
	3 -> ?WH('Accept-Encoding',"*",Params);
	_ when list(Enc) ->
	    ?WH('Accept-Encoding',Enc,Params)
    end.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%	%%
%% Accept-Language: * | <lang> [q=<q-value>]
%% Type: Multiple
%% Ref: 8.4.2.10
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_accept_language(Value) ->
    ?WH('Accept-Language',Value,[]).

format_accept_language(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_accept_language(H, _Version) ->
    case H#wsp_header.value of
	"*" -> ?ENCODE_SHORT(0);
	Lang -> case catch encode_lang(Lang) of
		    {'EXIT', _} -> encode_text_string(Lang);
		    Code -> encode_integer(Code)
		end
    end.

decode_accept_language(0, _Version) ->
    ?WH('Accept-Language',"*",[]);
decode_accept_language(Value, _Version) when integer(Value) ->
    ?WH('Accept-Language',decode_lang(Value),[]);
decode_accept_language(Value, _Version) when list(Value) ->
    ?WH('Accept-Language',Value,[]);
decode_accept_language({_,Data}, _Version) ->
    {Data1, QData} = scan_header_data(Data),
    Charset = case Data1 of
		  0 ->
		      "*";
		  Value1 when integer(Value1) ->
		      decode_lang(Value1);
		  Value1 when list(Value1) ->
		      Value1;
		  {short,Data2} ->
		      decode_lang(d_long(Data2))
	      end,
    Params =
	if QData == <<>> ->
		[];
	   true ->
		{QVal,_} = d_q_value(QData),
		[{q, QVal}]
	end,
    ?WH('Accept-Language',Charset,Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Accept-Ranges: none | bytes | <extension>
%% Type: single
%% Ref:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_accept_ranges(Value) ->
    ?WH('Accept-Ranges', Value, []).

format_accept_ranges(H) ->
    H#wsp_header.value.

encode_accept_ranges(H, _Version) ->
    case H#wsp_header.value of
	"none" -> ?ENCODE_SHORT(0);
	"bytes" -> ?ENCODE_SHORT(1);
	Value -> encode_text_string(Value)
    end.

decode_accept_ranges(0, _Version) ->
    ?WH('Accept-Ranges', "none", []);
decode_accept_ranges(1, _Version) ->
    ?WH('Accept-Ranges', "bytes", []);
decode_accept_ranges(Value, _Version) when list(Value) ->
    ?WH('Accept-Ranges', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Age: <delta-seconds>
%% Type: single
%% Ref:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_age(Value) ->
    %% FIXME
    ?WH('Age', Value, []).

format_age(H) ->
    integer_to_list(H#wsp_header.value).

encode_age(H, _Version) ->
    e_delta_seconds(H#wsp_header.value).

decode_age(Value, _Version) when integer(Value) ->
    ?WH('Age', Value, []);
decode_age({short,Data}, _Version) ->
    ?WH('Age', d_long(Data), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Allow: <well-known-method>
%% Type: multiple
%% Ref:
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_allow(Value) ->
    ?WH('Allow', parse_well_known_method(Value), []).

format_allow(H) ->
    atom_to_list(H#wsp_header.value).

encode_allow(H, Version) ->
    encode_well_known_method(H#wsp_header.value, Version).

decode_allow(Value, Version) ->
    ?WH('Allow', decode_well_known_method(Value,Version), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Authorization:
%% Ref: 8.4.2.14
%% Type: server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_authorization(Value) ->
    parse_credentials('Authorization', Value).

format_authorization(H) ->
    format_credentials(H#wsp_header.value, H#wsp_header.params).

encode_authorization(H, Version) ->
    encode_credentials(H#wsp_header.value, H#wsp_header.params, Version).

decode_authorization({_,Data}, Version) ->
    decode_credentials('Authorization', Data, Version).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%%
%% Cache-Control:
%% 8.4.2.15
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_cache_control(Value) ->
    case Value of
	"no-cache" -> ?WH('Cache-Control',Value,[]);
	"no-store" -> ?WH('Cache-Control',Value,[]);
	"max-stale" -> ?WH('Cache-Control',Value,[]);
	"only-if-cached" -> ?WH('Cache-Control',Value,[]);
	"private" -> ?WH('Cache-Control',Value,[]);
	"public" -> ?WH('Cache-Control',Value,[]);
	"no-transform" -> ?WH('Cache-Control',Value,[]);
	"must-revalidate" -> ?WH('Cache-Control',Value,[]);
	"proxy-revalidate" -> ?WH('Cache-Control',Value,[]);
	_ ->
	    Params = parse_params([Value]),
	    ?WH('Cache-Control',"",Params)
    end.

format_cache_control(H) ->
    if H#wsp_header.value == "" ->
	    format_params0(H#wsp_header.params);
       true ->
	    [H#wsp_header.value, format_params(H#wsp_header.params)]
    end.



encode_cache_control(H, Version) ->
    case H#wsp_header.value of
	"no-cache" -> ?ENCODE_SHORT(0);
	"no-store" -> ?ENCODE_SHORT(1);
	"max-stale" -> ?ENCODE_SHORT(3);
	"only-if-cached" -> ?ENCODE_SHORT(5);
	"private" -> ?ENCODE_SHORT(7);
	"public" -> ?ENCODE_SHORT(6);
	"no-transform" -> ?ENCODE_SHORT(8);
	"must-revalidate" -> ?ENCODE_SHORT(9);
	"proxy-revalidate" -> ?ENCODE_SHORT(10);
	"" ->
	    case H#wsp_header.params of
		[{'no-cache',Field}] ->
		    e_value(?ENCODE_SHORT(0),
			    e_field_name(Field,Version));
		[{'max-age',Sec}] ->
		    e_value(?ENCODE_SHORT(2),
			    e_delta_seconds(Sec));
		[{'max-fresh',Sec}] ->
		    e_value(?ENCODE_SHORT(4),
			    e_delta_seconds(Sec));
		[{'private',Field}] ->
		    e_value(?ENCODE_SHORT(7),
			    e_field_name(Field,Version));
		[{'s-maxage',Sec}] ->
		    e_value(?ENCODE_SHORT(11),
			    e_delta_seconds(Sec))
	    end;
	Ext ->
	    [Param] = H#wsp_header.params,
	    e_value(encode_text_string(Ext),
		    encode_parameter(Param, Version))
    end.


decode_cache_control(Value, _Version) when integer(Value) ->
    case Value of
	0 -> ?WH('Cache-Control',"no-cache",[]);
	1 -> ?WH('Cache-Control',"no-store",[]);
	3 -> ?WH('Cache-Control',"max-stale",[]);
	5 -> ?WH('Cache-Control',"only-if-cached",[]);
	7 -> ?WH('Cache-Control',"private",[]);
	6 -> ?WH('Cache-Control',"public",[]);
	8 -> ?WH('Cache-Control',"no-transform",[]);
	9 -> ?WH('Cache-Control',"must-revalidate",[]);
	10 -> ?WH('Cache-Control',"proxy-revalidate",[])
    end;
decode_cache_control(Value, _Version) when list(Value) ->
    ?WH('Cache-Control',Value,[]);
decode_cache_control({_,Data},Version) ->
    {CacheDir, Data1} = scan_header_data(Data),
    case CacheDir of
	0 ->
	    {Field,_} = d_field_name(Data1),
	    ?WH('Cache-Control',"",[{'no-cache',Field}]);
	2 ->
	    {Sec,_} = d_integer_value(Data1),
	    ?WH('Cache-Control',"",[{'max-age',Sec}]);
	4 ->
	    {Sec,_} = d_integer_value(Data1),
	    ?WH('Cache-Control',"",[{'max-fresh',Sec}]);
	7 ->
	    {Field,_} = d_field_name(Data1),
	    ?WH('Cache-Control',"",[{private,Field}]);
	11 ->
	    {Sec,_} = d_integer_value(Data1),
	    ?WH('Cache-Control',"",[{'s-maxage',Sec}]);
	Ext when list(Ext) ->
	    {Param,_} = decode_parameter(Data1, Version),
	    ?WH('Cache-Control',Ext,[Param])
    end.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Connection: close | Ext
%% Type: single
%% Ref: 8.4.2.16
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_connection(Value) ->
    ?WH('Connection', Value, []).

format_connection(H) ->
    H#wsp_header.value.

encode_connection(H, _Version) ->
    case H#wsp_header.value of
	"close" -> ?ENCODE_SHORT(0);
	Value ->  encode_text_string(Value)
    end.

decode_connection(0, _Version) ->
    ?WH('Connection', "close", []);
decode_connection(Value, _Version) when list(Value) ->
    ?WH('Connection', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Base: <uri>
%% Type: single
%% Ref:
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_content_base(Value) ->
    ?WH('Content-Base', Value, []).

format_content_base(H) ->
    H#wsp_header.value.

encode_content_base(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_content_base(Value, _Version) when list(Value) ->
    ?WH('Content-Base', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Encoding:
%% Ref: 8.4.2.18
%% Type: single
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_content_encoding(Value) ->
    ?WH('Content-Encoding', tolower(Value), []).

format_content_encoding(H) ->
    H#wsp_header.value.

encode_content_encoding(H, _Version) ->
    case H#wsp_header.value of
	"gzip"     -> ?ENCODE_SHORT(0);
	"compress" -> ?ENCODE_SHORT(1);
	"deflate"  -> ?ENCODE_SHORT(2);
	Value -> encode_text_string(Value)
    end.

decode_content_encoding(0, _Version) ->
    ?WH('Content-Encoding', "gzip", []);
decode_content_encoding(1, _Version) ->
    ?WH('Content-Encoding', "compress", []);
decode_content_encoding(2, _Version) ->
    ?WH('Content-Encoding',"deflate", []);
decode_content_encoding(Value, _Version) when list(Value) ->
    ?WH('Content-Encoding', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Language:
%% Ref: 8.4.2.19
%% Type: single
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_content_language(Value) ->
    ?WH('Content-Language', Value, []).

format_content_language(H) ->
    H#wsp_header.value.

encode_content_language(H, _Version) ->
    case H#wsp_header.value of
	"*" -> ?ENCODE_SHORT(0);
	Lang -> case catch encode_lang(Lang) of
		    {'EXIT', _} -> encode_text_string(Lang);
		    Code -> encode_integer(Code)
		end
    end.

decode_content_language(0, _Version) ->
    ?WH('Content-Language',"*",[]);
decode_content_language(Value, _Version) when integer(Value) ->
    ?WH('Content-Language',decode_lang(Value),[]);
decode_content_language(Value, _Version) when list(Value) ->
    ?WH('Content-Language',Value,[]);
decode_content_language({short,Data}, _Version) ->
    Value = d_long(Data),
    ?WH('Content-Language',decode_lang(Value),[]);
decode_content_language(Value, _Version) when list(Value) ->
    ?WH('Content-Language',Value,[]).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Length: <integer-value>
%% Ref: 8.4.2.20
%% Type: single
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_content_length(Value) ->
    ?WH('Content-Length', list_to_integer(Value), []).

format_content_length(H) ->
    integer_to_list(H#wsp_header.value).

encode_content_length(H, _Version) ->
    encode_integer(H#wsp_header.value).

decode_content_length(Value, _Version) when integer(Value) ->
    ?WH('Content-Length', Value, []);
decode_content_length({short,Data}, _Version) ->
    Value = d_long(Data),
    ?WH('Content-Length', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Location: <uri-value>
%% Ref: 8.4.2.21
%% Type: single
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_content_location(Value) ->
    ?WH('Content-Location', Value, []).

format_content_location(H) ->
    H#wsp_header.value.

encode_content_location(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_content_location(Value, _Version) when list(Value) ->
    ?WH('Content-Location', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Md5: <value-length> <digest>
%% Ref: 8.4.2.22
%% Type: single, end-to-end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_content_md5(Value) ->
    ?WH('Content-Md5', base64:decode(Value), []).

format_content_md5(H) ->
    base64:encode(H#wsp_header.value).

encode_content_md5(H, _Version) ->
    e_value(H#wsp_header.value).

decode_content_md5({_,Data}, _Version) ->
    ?WH('Content-Md5', Data, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Range: <first-byte-pos> <entity-len>
%% Ref: 8.4.2.23
%% Type: single
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_content_range(Value) ->
    %% FIXME:
    ?WH('Content-Range', Value, []).

format_content_range(H) ->
    {Pos,Len} = H#wsp_header.value,
    if Len == "*" ->
	    ["bytes ", integer_to_list(Pos), "-*/*"];
       true ->
	    ["bytes ", integer_to_list(Pos),"-",integer_to_list(Len-1),
	     "/", integer_to_list(Len)]
    end.

encode_content_range(H, _Version) ->
    case H#wsp_header.value of
	{Pos, "*"} ->
	    e_value(e_uintvar(Pos), <<128>>);
	{Pos, Len} ->
	    e_value(e_uintvar(Pos), e_uintvar(Len))
    end.

decode_content_range({_, Data}, _Version) ->
    {Pos, Data1} = d_uintvar(Data),
    Len =
	case Data1 of
	    <<128>> -> "*";
	    _ ->
		{L, _} = d_uintvar(Data1),
		L
	end,
    ?WH('Content-Range', {Pos,Len}, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Type:
%% Ref: 8.4.2.24
%% Type: single, end-to-end
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_content_type(Value) ->
    case string:tokens(Value, ";") of
	[Type | Ps] ->
	    Params = parse_params(Ps),
	    ?WH('Content-Type', Type, Params);
	[] ->
	    ?WH('Content-Type', Value, [])
    end.

format_content_type(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_content_type(H, Version) ->
    case encode_params(H#wsp_header.params,Version) of
	<<>> ->
	    encode_well_known_media(H#wsp_header.value, Version);
	Params ->
	    Media = encode_well_known_media(H#wsp_header.value, Version),
	    e_value(Media, Params)
    end.

decode_content_type(Value,Version) when integer(Value) ->
    ?WH('Content-Type', decode_well_known_media(Value,Version), []);
decode_content_type(Value,Version) when list(Value) ->
    ?WH('Content-Type', decode_well_known_media(Value,Version), []);
decode_content_type({_, Data}, Version) ->
    {Value,Data1} = scan_header_data(Data),
    ContentType = if integer(Value) ->
			  decode_well_known_media(Value,Version);
		     list(Value) ->
			  decode_well_known_media(Value,Version);
		     true ->
			  {_,Data2} = Value,
			  decode_well_known_media(d_long(Data2),Version)
		  end,
    Params = decode_params(Data1, Version),
    ?WH('Content-Type', ContentType, Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Date: <http-date>
%% Ref: 8.2.4.25
%% Type: single, end-to-end
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_date(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('Date', DateTime, []).

format_date(H) ->
    fmt_date(H#wsp_header.value).

encode_date(H, _Version) ->
    e_date(H#wsp_header.value).

decode_date(Value, _Version) ->
    ?WH('Date', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Etag: <text-string>
%% Ref: 8.2.4.26
%% Type: single, end-to-end
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_etag(Value) ->
    ?WH('Etag', Value, []).

format_etag(H) ->
    H#wsp_header.value.

encode_etag(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_etag(Value, _Version) ->
    ?WH('Etag', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Expires: <date-value>
%% Ref: 8.4.2.27
%% Type: single, end-to-end, server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_expires(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('Expires', DateTime, []).

format_expires(H) ->
    fmt_date(H#wsp_header.value).

encode_expires(H, _Version) ->
    e_date(H#wsp_header.value).

decode_expires(Value, _Version) ->
    ?WH('Expires', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% From: <text-string>
%% Ref: 8.4.2.28
%% Type: single,
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_from(Value) ->
    ?WH('From', Value, []).

format_from(H) ->
    H#wsp_header.value.

encode_from(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_from(Value, _Version) ->
    ?WH('From', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Host: <text-string>
%% Ref: 8.4.2.29
%% Type: single, end-to-end, client-to-server
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_host(Value) ->
    ?WH('Host', Value, []).

format_host(H) ->
    H#wsp_header.value.

encode_host(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_host(Value, _Version) ->
    ?WH('Host', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% If-Modified-Since: <date-value>
%% Ref: 8.4.2.30
%% Type: single, end-to-end, client-to-server
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_if_modified_since(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('If-Modified-Since', DateTime, []).

format_if_modified_since(H) ->
    fmt_date(H#wsp_header.value).

encode_if_modified_since(H, _Version) ->
    e_date(H#wsp_header.value).

decode_if_modified_since(Value, _Version) ->
    ?WH('If-Modified-Since', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% If-Match: <text-string>
%% Ref: 8.4.2.31
%% Type: end-to-end, client-to-server
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_if_match(Value) ->
    ?WH('If-Match', Value, []).

format_if_match(H) ->
    H#wsp_header.value.

encode_if_match(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_if_match(Value, _Version) ->
    ?WH('If-Match', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% If-None-Match: <text-string>
%% Ref: 8.4.2.32
%% Type: end-to-end, client-to-server
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_if_none_match(Value) ->
    ?WH('If-None-Match', Value, []).

format_if_none_match(H) ->
    H#wsp_header.value.

encode_if_none_match(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_if_none_match(Value, _Version) ->
    ?WH('If-None-Match', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% If-Range: Text | Date
%% Ref: 8.4.2.33
%% Type: end-to-end, client-to-server
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_if_range(Value) ->
    case catch parse_http_date(Value) of
	{'EXIT', _} ->
	    ?WH('If-Range', Value, []);
	{DateTime,_} ->
	    ?WH('If-Range', DateTime, [])
    end.


format_if_range(H) ->
    case H#wsp_header.value of
	Value when list(Value) -> Value;
	DateTime -> fmt_date(DateTime)
    end.

encode_if_range(H, _Version) ->
    case H#wsp_header.value of
	Value when list(Value) ->
	    encode_text_string(Value);
	DateTime ->
	    e_date(DateTime)
    end.

decode_if_range(Value, _Version) when list(Value) ->
    ?WH('If-Range', decode_text_string(Value), []);
decode_if_range(Value, _Version) ->
    ?WH('If-Range', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% If-Unmodified-Since: <date-value>
%% Ref: 8.4.2.34
%% Type: single, end-to-end, client-to-server
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_if_unmodified_since(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('If-Unmodified-Since', DateTime, []).

format_if_unmodified_since(H) ->
    fmt_date(H#wsp_header.value).

encode_if_unmodified_since(H, _Version) ->
    e_date(H#wsp_header.value).

decode_if_unmodified_since(Value, _Version) ->
    ?WH('If-Unmodified-Since', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Location: <uri-value>
%% Ref: 8.4.2.36
%% Type: single, end-to-end, server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_location(Value) ->
    ?WH('Location', Value, []).

format_location(H) ->
    H#wsp_header.value.

encode_location(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_location(Value, _Version) when list(Value) ->
    ?WH('Location', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Last-Modified: <date-value>
%% Ref: 8.4.2.35
%% Type: single, end-to-end, server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_last_modified(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('Last-Modified', DateTime, []).

format_last_modified(H) ->
    fmt_date(H#wsp_header.value).

encode_last_modified(H, _Version) ->
    e_date(H#wsp_header.value).

decode_last_modified(Value, _Version) ->
    ?WH('Last-Modified', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Max-Forwards: <integer-value>
%% Ref: 8.4.2.37
%% Type: single, end-to-end, server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_max_forwards(String) ->
    ?WH('Max-Forwards', list_to_integer(String), []).

format_max_forwards(H) ->
    integer_to_list(H#wsp_header.value).

encode_max_forwards(H, _Version) ->
    encode_integer(H#wsp_header.value).

decode_max_forwards(Value, _Version) ->
    decode_integer(Value).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Pragma: No-Cache | value-length Parameter
%% Ref:
%% Type:
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_pragma(Value) ->
    ?WH('Pragma',Value,[]).

format_pragma(H) ->
    case H#wsp_header.value of
	"" -> format_params(H#wsp_header.params);
	Value -> Value
    end.

encode_pragma(H, Version) ->
    case H#wsp_header.value of
	"no-cache" -> ?ENCODE_SHORT(0);
	"" ->
	    encode_parameter(hd(H#wsp_header.params), Version)
    end.

decode_pragma(0, _Version) ->
    ?WH('Pragma',"no-cache",[]);
decode_pragma({_,Data}, Version) ->
    {Param,_} = decode_parameter(Data, Version),
    ?WH('Pragma',"",[Param]).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Proxy-Authenticate:
%% Ref: 8.4.2.39
%% Type: single?, client-to-proxy
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_proxy_authenticate(Value) ->
    parse_challenge('Proxy-Authenticate', Value).

format_proxy_authenticate(H) ->
    format_challenge(H#wsp_header.value, H#wsp_header.params).

encode_proxy_authenticate(H, Version) ->
    encode_challenge(H#wsp_header.value,
		     H#wsp_header.params, Version).

decode_proxy_authenticate({_, Data}, Version) ->
    decode_challenge('Proxy-Authenticate', Data, Version).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Proxy-authorization:
%% Ref: 8.4.2.40
%% Type: single?, proxy-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_proxy_authorization(Value) ->
    parse_credentials('Proxy-Authorization', Value).

format_proxy_authorization(H) ->
    format_credentials(H#wsp_header.value, H#wsp_header.params).

encode_proxy_authorization(H, Version) ->
    encode_credentials(H#wsp_header.value, H#wsp_header.params, Version).

decode_proxy_authorization({_,Data}, Version) ->
    decode_credentials('Proxy-Authorization', Data, Version).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Public: <well-known-method> | Token-Text
%% Ref: 8.4.2.41
%% Type:
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_public(Value) ->
    ?WH('Public', parse_well_known_method(Value), []).

format_public(H) ->
    if atom(H#wsp_header.value) ->
	    atom_to_list(H#wsp_header.value);
       list(H#wsp_header.value) ->
	    H#wsp_header.value
    end.

encode_public(H, Version) ->
    if atom(H#wsp_header.value) ->
	    encode_well_known_method(H#wsp_header.value,Version);
       list(H#wsp_header.value) ->
	    encode_text_string(H#wsp_header.value)
    end.

decode_public(Value, _Version) when list(Value) ->
    ?WH('Public', Value, []);
decode_public(Value, Version) ->
    ?WH('Public', decode_well_known_method(Value,Version), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Range:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_range(Value) ->
    %% FIXME:
    ?WH('Range', Value, []).

format_range(H) ->
    case H#wsp_header.value of
	{First,undefined} ->
	    ["bytes=", integer_to_list(First), "-"];
	{First,Last} ->
	    ["bytes=", integer_to_list(First), "-", integer_to_list(Last)];
	Len when integer(Len) ->
	    ["bytes=-", integer_to_list(Len)]
    end.

encode_range(H, _Version) ->
    case H#wsp_header.value of
	{First,undefined} ->
	    e_value(?ENCODE_SHORT(0),
		    e_uintvar(First));
	{First,Last} ->
	    e_value(?ENCODE_SHORT(0),
		    e_uintvar(First),
		    e_uintvar(Last));
	Len when integer(Len) ->
	    e_value(?ENCODE_SHORT(1),
		    e_uintvar(Len))
    end.

decode_range({_,Data}, _Version) ->
    case scan_header_data(Data) of
	{0, Data1} ->
	    case d_uintvar(Data1) of
		{First, <<>>} ->
		    ?WH('Range', {First, undefined},[]);
		{First, Data2} ->
		    {Last, _} = d_uintvar(Data2),
		    ?WH('Range', {First, Last}, [])
	    end;
	{1, Data1} ->
	    {Len, _} =d_uintvar(Data1),
	    ?WH('Range', Len, [])
    end.


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Referer: <uri-value>
%% Ref: 8.4.2.43
%% Type: single
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_referer(Value) ->
    ?WH('Referer', Value, []).

format_referer(H) ->
    H#wsp_header.value.

encode_referer(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_referer(Value, _Version) when list(Value) ->
    ?WH('Referer', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Retry-After: Value-length (Retry-date-value | Retry-delta-seconds)
%% Ref: 8.4.2.44
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_retry_after(Value) ->
    case catch parse_http_date(Value) of
	{'EXIT', _} ->
	    ?WH('Retry-After', list_to_integer(Value), []);
	{DateTime,_} ->
	    ?WH('Retry-After', DateTime, [])
    end.

format_retry_after(H) ->
    Value = H#wsp_header.value,
    if integer(Value) ->
	    integer_to_list(Value);
       true ->
	    fmt_date(Value)
    end.

encode_retry_after(H, _Version) ->
    Value = H#wsp_header.value,
    if integer(Value) ->
	    e_value(?ENCODE_SHORT(1),
		    e_delta_seconds(Value));
       true ->
	    e_value(?ENCODE_SHORT(0),
		    e_date(Value))
    end.

decode_retry_after({_,Data}, _Version) ->
    case scan_header_data(Data) of
	{0, Data1} ->
	    ?WH('Retry-After', d_date(Data1), []);
	{1, Data1} ->
	    case scan_header_data(Data1) of
		Sec when integer(Sec) ->
		    ?WH('Retry-After', Sec, []);
		{short,Data2} ->
		    ?WH('Retry-After', d_long(Data2), [])
	    end
    end.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Server: <text-string>
%% Ref: 8.4.2.45
%% Type: server-to-client
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_server(Value) ->
    ?WH('Server', Value, []).

format_server(H) ->
    H#wsp_header.value.

encode_server(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_server(Value, _Version) ->
    ?WH('Server', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Transfer-Encoding:
%% Ref: 8.4.2.46
%% Type: hop-by-hop
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_transfer_encoding(Value) ->
    ?WH('Transfer-Encoding', Value, []).

format_transfer_encoding(H) ->
    H#wsp_header.value.

encode_transfer_encoding(H, _Version) ->
    case H#wsp_header.value of
	"chunked" -> ?ENCODE_SHORT(0);
	Value -> encode_text_string(Value)
    end.

decode_transfer_encoding(0, _Version) ->
    ?WH('Transfer-Encoding', "chunked", []);
decode_transfer_encoding(Value, _Version) when list(Value)->
    ?WH('Transfer-Encoding', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Upgrade: Text-String
%% Ref: 8.4.2.47
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_upgrade(Value) ->
    ?WH('Upgrade', Value, []).

format_upgrade(H) ->
    H#wsp_header.value.

encode_upgrade(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_upgrade(Value, _Version) when list(Value) ->
    ?WH('Upgrade', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% User-Agent:
%% Ref: 8.4.2.48
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_user_agent(Value) ->
    ?WH('User-Agent', Value, []).

format_user_agent(H) ->
    H#wsp_header.value.

encode_user_agent(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_user_agent(Value, _Version) ->
    ?WH('User-Agent', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Vary: Well-known-header-field | Token-text
%% Ref: 8.4.2.49
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_vary(Value) ->
    ?WH('Vary', normalise_field_name(Value), []).

format_vary(H) ->
    to_list(H#wsp_header.value).

encode_vary(H, Version) ->
    e_field_name(H#wsp_header.value, Version).

decode_vary(Value, _Version) when integer(Value) ->
    ?WH('Vary', lookup_field_name(Value), []);
decode_vary(Value, _Version) when list(Value) ->
    ?WH('Vary', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Via: <text-string>
%% Ref: 8.4.2.50
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_via(Value) ->
    ?WH('Via', Value, []).

format_via(H) ->
    H#wsp_header.value.

encode_via(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_via(Value, _Version) when list(Value) ->
    ?WH('Via', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Warning: Warn-Code | Warning-value
%% Ref: 8.4.2.51
%% Type: general, multiple
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_warning(Value) ->
    case string:tokens(Value, " ") of
	[Code] ->
	    ?WH('Warning', {list_to_integer(Code),"",""}, []);
	[Code,Agent,Text] ->
	    ?WH('Warning', {list_to_integer(Code), Agent, Text}, [])
    end.

format_warning(H) ->
    case H#wsp_header.value of
	{Code, "", ""} ->
	    integer_to_list(Code);
	{Code, Agent, Text} ->
	    [integer_to_list(Code), " ", Agent, " ", Text]
    end.

encode_warning(H, _Version) ->
    case H#wsp_header.value of
	{Code,"",""} ->
	    ?ENCODE_SHORT(Code);
	{Code, Agent, Text} ->
	    e_value(?ENCODE_SHORT(Code),
		    encode_text_string(Agent),
		    encode_text_string(Text))
    end.

decode_warning(Value, _Version) when integer(Value) ->
    ?WH('Warning', {Value, "", ""}, []);
decode_warning({_, Data}, _Version) ->
    {Code,Data1}= scan_header_data(Data),
    {Agent,Data2} = d_text_string(Data1),
    {Text,_Data3} = d_text_string(Data2),
    ?WH('Warning', {Code,Agent,Text}, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% WWW-Authenticate: challenge
%% Ref: 8.4.2.52
%% Type: single? client-to-server
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_www_authenticate(Value) ->
    parse_challenge('Www-Authenticate', Value).

format_www_authenticate(H) ->
    format_challenge(H#wsp_header.value, H#wsp_header.params).

encode_www_authenticate(H, Version) ->
    encode_challenge(H#wsp_header.value,
		     H#wsp_header.params, Version).

decode_www_authenticate({_, Data}, Version) ->
    decode_challenge('Www-Authenticate', Data, Version).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Disposition: "form-data" | "attachment" [<param>]*
%% Ref: 8.4.2.53
%% Type: single
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_content_disposition(Value) ->
    ?WH('Content-Disposition', Value, []).

format_content_disposition(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_content_disposition(H, Version) ->
    case H#wsp_header.value of
	"form-data" ->
	    e_value(?ENCODE_SHORT(0),
		    encode_params(H#wsp_header.params, Version));
	"attachment" ->
	    e_value(?ENCODE_SHORT(1),
		    encode_params(H#wsp_header.params, Version))
    end.

decode_content_disposition({_,Data}, Version) when binary(Data) ->
    case scan_header_data(Data) of
	{0, Data1} ->
	    Params = decode_params(Data1, Version),
	    ?WH('Content-Disposition', "form-data", Params);
	{1, Data1} ->
	    Params = decode_params(Data1, Version),
	    ?WH('Content-Disposition', "attachment", Params)
    end.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Application-Id:
%% Ref: 8.4.2.54
%% Type:
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_application_id(Value) ->
    ?WH('X-Wap-Application-Id', Value, []).

format_x_wap_application_id(H) ->
    H#wsp_header.value.

encode_x_wap_application_id(H, _Version) ->
    encode_push_application(H#wsp_header.value).

decode_x_wap_application_id(Value, _Version) ->
    ?WH('X-Wap-Application-Id', decode_push_application(Value),[]).


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Content-Uri: <uri-value>
%% Ref: 8.4.2.55
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_content_uri(Value) ->
    ?WH('X-Wap-Content-Uri', Value, []).

format_x_wap_content_uri(H) ->
    H#wsp_header.value.

encode_x_wap_content_uri(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_x_wap_content_uri(Value, _Version) when list(Value) ->
    ?WH('X-Wap-Content-Uri', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Initiator-Uri: <uri-value>
%% Ref: 8.4.2.56
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_initiator_uri(Value) ->
    ?WH('X-Wap-Initiator-Uri', Value, []).

format_x_wap_initiator_uri(H) ->
    H#wsp_header.value.

encode_x_wap_initiator_uri(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_x_wap_initiator_uri(Value, _Version) when list(Value) ->
    ?WH('X-Wap-Initiator-Uri', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Accept-Application: Any-Application | Appication-Id-Value
%% Ref: 8.4.2.57
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_accept_application(Value) ->
    ?WH('Accept-Application', Value, []).

format_accept_application(H) ->
    H#wsp_header.value.


encode_accept_application(H, _Version) ->
    case H#wsp_header.value of
	"*" -> ?ENCODE_SHORT(0);
	Value ->
	    case catch encode_push_application(Value) of
		{'EXIT',_} ->
		    encode_uri_value(Value);
		App ->
		    encode_integer(App)
	    end
    end.

decode_accept_application(0, _Version) ->
    ?WH('Accept-Application', "*", []);
decode_accept_application(Value, _Version) when integer(Value) ->
    ?WH('Accept-Application', decode_push_application(Value), []);
decode_accept_application({short,Data}, _Version) ->
    Value = d_long(Data),
    ?WH('Accept-Application', decode_push_application(Value), []);
decode_accept_application(Value, _Version) when list(Value) ->
    ?WH('Accept-Application', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Bearer-Indication: <integer-value>
%% Type: sinlge
%% Ref: 8.4.2.58
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_bearer_indication(Value) ->
    ?WH('Bearer-Indication', Value, []).

format_bearer_indication(H) ->
    integer_to_list(H#wsp_header.value).

encode_bearer_indication(H, _Version) ->
    encode_integer(H#wsp_header.value).

decode_bearer_indication(Value, _Version) when integer(Value) ->
    ?WH('Bearer-Indication', Value, []);
decode_bearer_indication({short,Data}, _Version) ->
    Value = d_long(Data),
    ?WH('Bearer-Indication', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Push-Flag: Short-Integer
%% Type: single
%% Ref: 8.4.2.59
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_push_flag(Value) ->
    ?WH('Push-Flag', integer_to_list(Value), []).

format_push_flag(H) ->
    integer_to_list(H#wsp_header.value).

encode_push_flag(H, _Version) ->
    ?ENCODE_SHORT(H#wsp_header.value).

decode_push_flag(Value, _Version) when integer(Value) ->
    ?WH('Push-Flag', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Profile: <uri-value>
%% Ref: 8.4.2.60
%% Type: single, hop-by-hop, client-to-proxy
%%
%% Note: Normally transfered as 'X-Wap-Profile'
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_profile(Value) ->
    ?WH('Profile', Value, []).

format_profile(H) ->
    H#wsp_header.value.

encode_profile(H, _Version) ->
    encode_uri_value(H#wsp_header.value).

decode_profile(Value, _Version) ->
    ?WH('Profile', decode_uri_value(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Profile-Diff: Value-Length Octets
%% Ref: 8.4.2.61
%% Type: single, hop-by-hop, client-to-proxy
%%
%%  Value is WBXML encoded profile diff information
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_profile_diff(Value) ->
    %% FIXME parse XML code?
    ?WH('Profile-Diff', Value, []).

format_profile_diff(_H) ->
    %% FIXME emit ???
    "WBXML".

encode_profile_diff(H, _Version) ->
    e_value(H#wsp_header.value).

decode_profile_diff({_,Value}, _Version) ->
    ?WH('Profile-Diff', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Profile-Warning: Code
%% Ref: 8.4.2.62
%% Type: single
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_profile_warning(Value) ->
    ?WH('Profile-Warning', {Value,"",undefined}, []).

format_profile_warning(H) ->
    {Code,Target,Date} = H#wsp_header.value,
    CodeData = integer_to_list(Code),
    if Target == "", Date == undefined ->
	    CodeData;
       Date == undefined ->
	    [CodeData," ",Target];
       true ->
	    [CodeData," ",Target," ",format_date(Date)]
    end.


encode_profile_warning(H, _Version) ->
    {Code,Target,Date} = H#wsp_header.value,
    CodeData = case Code of
		   100 -> ?ENCODE_SHORT(16#10);
		   101 -> ?ENCODE_SHORT(16#11);
		   102 -> ?ENCODE_SHORT(16#12);
		   200 -> ?ENCODE_SHORT(16#20);
		   201 -> ?ENCODE_SHORT(16#21);
		   202 -> ?ENCODE_SHORT(16#22);
		   203 -> ?ENCODE_SHORT(16#23)
	       end,
    if Target == "", Date == undefined ->
	    CodeData;
       Date == undefined ->
	    e_value(CodeData, encode_text_string(Target));
       true ->
	    e_value(CodeData, encode_text_string(Target), e_date(Date))
    end.


decode_profile_warning(Value, _Version) when integer(Value) ->
    Code = case Value of
	       16#10 -> 100;
	       16#11 -> 101;
	       16#12 -> 102;
	       16#20 -> 200;
	       16#21 -> 201;
	       16#22 -> 202;
	       16#23 -> 203
	   end,
    ?WH('Profile-Warning', {Code,"",undefined}, []);
decode_profile_warning({_, <<1:1, Value:7, Data>>}, _Version) ->
    Code = case Value of
	       16#10 -> 100;
	       16#11 -> 101;
	       16#12 -> 102;
	       16#20 -> 200;
	       16#21 -> 201;
	       16#22 -> 202;
	       16#23 -> 203
	   end,
    {Target,Data1} = d_text_string(Data),
    Date =
	if Data1 == <<>> ->
		undefined;
	   true ->
		{DateValue,_} = scan_header_data(Data1),
		d_date(DateValue)
	end,
    ?WH('Profile-Warning', {Code,Target,Date}, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Expect: 100-contine | Expect-expression
%% Ref: 8.4.2.63
%% Type: client-to-server
%% Note: Bug in the spec value-length is missing !!!
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_expect(Value) ->
    ?WH('Expect', Value, []).

format_expect(H) ->
    case H#wsp_header.value of
	{Var,Val} ->
	    [Var,"=",Val, format_params(H#wsp_header.params)];
	Val when list(Val) ->
	    Val
    end.

encode_expect(H, Version) ->
    case H#wsp_header.value of
	"100-continue" ->
	    ?ENCODE_SHORT(0);
	{Var,Val} ->
	    e_value(encode_text_string(Var),
		    encode_text_string(Val),
		    encode_params(H#wsp_header.params,Version))
    end.

decode_expect(0, _Version) ->
    ?WH('Expect', "100-continue", []);
decode_expect({_, Data}, Version) ->
    {Var, Data1} = d_text_string(Data),
    {Val, Data2} = d_text_string(Data1),
    Params = decode_params(Data2, Version),
    ?WH('Expect', {decode_text_string(Var),
		   decode_text_string(Val)}, Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Te: Trailers | TE-General-From
%% Ref: 8.4.2.64
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_te(Value) ->
    ?WH('Te', Value, []).

format_te(H) ->
    [H#wsp_header.value, format_params(H#wsp_header.params)].

encode_te(H, Version) ->
    case H#wsp_header.value of
	"trailers" -> ?ENCODE_SHORT(1);
	"chunked" ->
	    e_value(?ENCODE_SHORT(2),
		    encode_params(H#wsp_header.params,Version));
	"identity" ->
	    e_value(?ENCODE_SHORT(3),
		    encode_params(H#wsp_header.params,Version));
	"gzip" ->
	    e_value(?ENCODE_SHORT(4),
		    encode_params(H#wsp_header.params,Version));
	"compress" ->
	    e_value(?ENCODE_SHORT(5),
		    encode_params(H#wsp_header.params,Version));
	"deflate" ->
	    e_value(?ENCODE_SHORT(6),
		    encode_params(H#wsp_header.params,Version));
	Value ->
	    e_value(encode_text_string(Value),
		    encode_params(H#wsp_header.params,Version))
    end.

decode_te(1, _Version) ->
    ?WH('Te', "trailers", []);
decode_te({_, Data}, _Version) ->
    {Val, Data1} = scan_header_data(Data),
    Value =
	case Val of
	    2 -> "chunked";
	    3 -> "identity";
	    4 -> "gzip";
	    5 -> "compress";
	    6 -> "deflate";
	    V when list(V) -> V
	end,
    Params = case Data1 of
		 <<>> ->  [];
		 <<128, QData>> ->
		     {QValue, _} = d_q_value(QData),
		     [{q, QValue}]
	     end,
    ?WH('Te', Value, Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Trailer: Well-known-header-field | Token-text
%% Ref: 8.4.2.65
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_trailer(Value) ->
    ?WH('Trailer', normalise_field_name(Value), []).

format_trailer(H) ->
    to_list(H#wsp_header.value).

encode_trailer(H, Version) ->
    e_field_name(H#wsp_header.value, Version).

decode_trailer(Value, _Version) when integer(Value) ->
    ?WH('Trailer', lookup_field_name(Value), []);
decode_trailer(Value, _Version) when list(Value) ->
    ?WH('Trailer', Value, []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Tod:
%% Ref: 8.4.2.66
%% Type: hop-by-hop
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_tod(String) ->
    {DateTime, _} = parse_http_date(String),
    ?WH('X-Wap-Tod', DateTime, []).

format_x_wap_tod(H) ->
    fmt_date(H#wsp_header.value).

encode_x_wap_tod(H, _Version) ->
    e_date(H#wsp_header.value).

decode_x_wap_tod(Value, _Version) ->
    ?WH('X-Wap-Tod', d_date(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Content-Id: <quoted-string>
%% Type:
%% Ref: 8.4.2.67
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_content_id(Value) ->
    ?WH('Content-Id', Value, []).

format_content_id(H) ->
    [$", H#wsp_header.value, $"].

encode_content_id(H, _Version) ->
    encode_quoted_string(H#wsp_header.value).

decode_content_id(Value, _Version) when list(Value) ->
    ?WH('Content-Id', decode_quoted_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Set-Cookie: <len> <cookie-version> <cookie-name> <cokie-value> <parm>*
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_set_cookie(String) ->
    %% MEGA FIXME; Cookie-value may be a quoted string and
    %% contain both ,=; etc Fix several cookies on same line!!
    case string:tokens(String, ";") of
	[Cookie | Ps] ->
	    case string:tokens(Cookie, "=") of
		[Name,Value] ->
		    Params = parse_params(Ps),
		    ?WH('Set-Cookie', {{1,0}, Name, Value}, Params);
		[Name] ->
		    Params = parse_params(Ps),
		    ?WH('Set-Cookie', {{1,0}, Name, ""}, Params)
	    end;
	[] ->
	    ?WH('Set-Cookie', {{1,0}, String, ""}, [])
    end.

format_set_cookie(H) ->
    case H#wsp_header.value of
	{{1,0},Name,Value} ->
	    [Name, "=", Value,format_params(H#wsp_header.params)];
	{Version,Name,Value} ->
	    [format_version(Version)," ",
	     Name, "=", Value,
	     format_params(H#wsp_header.params)]
    end.

encode_set_cookie(H, Version) ->
    {CookieVersion,Name,Value} = H#wsp_header.value,
    e_value(encode_version(CookieVersion),
	    encode_text_string(Name),
	    encode_text_string(Value),
	    encode_params(H#wsp_header.params, Version)).

decode_set_cookie({_, Data}, Version) ->
    {CookieVersion, Data1} = scan_header_data(Data),
    {CookieName, Data2} = scan_header_data(Data1),
    {CookieValue, Data3} = scan_header_data(Data2),
    Params = decode_params(Data3, Version),
    ?WH('Set-Cookie', {decode_version(CookieVersion),
		       decode_text_string(CookieName),
		       decode_text_string(CookieValue)}, Params).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Cookie:
%% Ref: 8.4.2.69
%% Type: single?, client-to-server
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_cookie(Value) ->
    %% FIXME parse cookie version etc
    ?WH('Cookie', {{1,0},Value}, []).

format_cookie(H) ->
    case H#wsp_header.value of
	{{1,0}, Cookies} ->
	    lists:map(fun({Name,Value,Ps}) ->
			      [Name,"=",Value, format_params(Ps)]
		      end, Cookies);
	{Version, Cookies} ->
	    [format_version(Version)," ",
	     lists:map(fun({Name,Value,Ps}) ->
			       [Name,"=",Value, format_params(Ps)]
		       end, Cookies)]
    end.

encode_cookie(H, Version) ->
    {Version, Cookies} = H#wsp_header.value,
    e_value(encode_version(Version),
	    encode_cookies(Cookies, [])).

encode_cookies([{Name,Value,Ps} | Cs], Acc) ->
    List =
	[encode_text_string(Name),
	 encode_text_string(Value) |
	 case Ps of
	     [{path,P},{domain,D}] ->
		 [encode_text_string(P), encode_text_string(D)];
	     [{domain,D},{path,P}] ->
		 [encode_text_string(P), encode_text_string(D)];
	     [{path,P}] ->
		 [encode_text_string(P)];
	     [{domain,D}] ->
		 [encode_text_string(""), encode_text_string(D)];
	     [] ->
		 []
	 end],
    Sz = lists:sum(lists:map(fun(B) -> size(B) end, List)),
    encode_cookies(Cs, [[e_uintvar(Sz) | List] | Acc]);
encode_cookies([], Acc) ->
    list_to_binary(lists:reverse(Acc)).


decode_cookie({_, Data}, _Version) ->
    {CookieVersion, Data1} = scan_header_data(Data),
    Cookies = decode_cookies(Data1, []),
    ?WH('Cookie', {decode_version(CookieVersion), Cookies}, []).

decode_cookies(<<>>, Acc) ->
    lists:reverse(Acc);
decode_cookies(Data0, _Acc) ->	%% IS IGNORING Acc A BUG OR NOT ?
    {Len, Data1} = d_uintvar(Data0),
    <<C0:Len/binary, Data2/binary>> = Data1,
    {Name, C1} = scan_header_data(C0),
    {Value, C2} = scan_header_data(C1),
    {Ps1, C3} =
	case d_text_string(C2) of
	    {"", C21} -> {[], C21};
	    {Path,C21} -> {[{path,Path}], C21}
	end,
    {Ps2, _} =
	case C3 of
	    <<>> -> {[], <<>>};
	    _ ->
		{Domain,C4} = d_text_string(C3),
		{[{domain,Domain}], C4}
	end,
    decode_cookies(Data2, [{decode_text_string(Name),
			    decode_text_string(Value),
			    Ps1++Ps2}]).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% Encoding-Version: Version-Value | Value-length Code-Page [Version-Value]
%% Ref: 8.4.2.70
%% Type: single, hop-by-hop, client-and-proxys
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_encoding_version(Value) ->
    ?WH('Encoding-Version', parse_version(Value), []).

format_encoding_version(H) ->
    format_version(H#wsp_header.value).

encode_encoding_version(H, _Version) ->
    encode_version(H#wsp_header.value).

decode_encoding_version(Value, _Version) when integer(Value) ->
    ?WH('Encoding-Version', decode_version(Value), []);
decode_encoding_version(Value, _Version) when list(Value) ->
    %% Note: in this case we parse the Value since we
    %% Must know the Encoding version
    ?WH('Encoding-Version', parse_version(Value), []);
decode_encoding_version({_,<<_:1,_CodePage:7>>}, _Version) ->
    %% ??? FIXME
    ?WH('Encoding-Version', "", []);
decode_encoding_version({_,<<_:1,_CodePage:7, Data1/binary>>}, _Version) ->
    {Value,_Data2} = scan_header_data(Data1),
    %% FIXME CodePage
    ?WH('Encoding-Version', decode_version(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Security:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_security(Value) ->
    ?WH('X-Wap-Security', Value, []).

format_x_wap_security(H) ->
    H#wsp_header.value.

encode_x_wap_security(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_x_wap_security(Value, _Version) ->
    ?WH('X-Wap-Security', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Loc-Invocation:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_loc_invocation(Value) ->
    ?WH('X-Wap-Loc-Invocation', Value, []).

format_x_wap_loc_invocation(H) ->
    H#wsp_header.value.

encode_x_wap_loc_invocation(H, _Version) ->
    encode_text_string(H#wsp_header.value).

decode_x_wap_loc_invocation(Value, _Version) ->
    ?WH('X-Wap-Loc-Invocation', decode_text_string(Value), []).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%
%% X-Wap-Loc-Delivery:
%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

parse_x_wap_loc_delivery(Value) ->
    ?WH('X-Wap-Loc-Delivery', Value, []).

format_x_wap_loc_delivery(H) ->
    H#wsp_header.value.

encode_x_wap_loc_delivery(H, _Value) ->
    encode_text_string(H#wsp_header.value).

decode_x_wap_loc_delivery(Value, _Version) ->
    ?WH('X-Wap-Loc-Delivery', decode_text_string(Value), []).


%%
%% Header Field parameters
%%

parse_params([Param|Ps]) ->
    case string:tokens(Param, "=") of
	[Name,Value0] ->
	    Val = trim(Value0),
	    P = case trim(tolower(Name)) of
		    "q" ->{q,Val};
		    "charset" -> {charset,Val};
		    "level" -> {level,Val};
		    "type" -> {type,Val};
		    "name" -> {name,Val};
		    "filename" -> {filename,Val};
		    "differences" -> {differences,Val};
		    "padding" -> {padding,Val};
		    "start" -> {start,Val};
		    "start-info" -> {'start-info',Val};
		    "comment" -> {comment,Val};
		    "domain" -> {domain,Val};
		    "max-age" -> {'max-age',Val};
		    "path" -> {path,Val};
		    "secure" -> {secure,no_value};
		    "sec" -> {sec, Val};
		    "mac" -> {mac, Val};
		    "creation-date" -> {'creation-date', Val};
		    "modification-date" -> {'modification-date', Val};
		    "read-date" -> {'read-date', Val};
		    "size" -> {size, Val};
		    Nm -> {Nm, Val}
		end,
	    [P | parse_params(Ps)];
	_ ->
	    parse_params(Ps)
    end;
parse_params([]) ->
    [].

%% format Params without leading ";"
format_params0([{Param,no_value}|Ps]) ->
    [to_list(Param) | format_params(Ps)];
format_params0([{Param,Value}|Ps]) ->
    [to_list(Param),"=",to_list(Value) | format_params(Ps)].

format_params(Ps) ->
    lists:map(fun({Param,no_value}) ->
		      ["; ", to_list(Param)];
		 ({Param,Value})->
		      ["; ", to_list(Param),"=",to_list(Value)]
	      end, Ps).


encode_params(Params, Version) ->
    list_to_binary(encode_params1(Params,Version)).

encode_params1([Param|Ps], Version) ->
    [ encode_parameter(Param, Version) | encode_params1(Ps, Version)];
encode_params1([], _Version) ->
    [].


decode_params(Data, Version) ->
    decode_params(Data, [], Version).

decode_params(<<>>, Ps, _Version) ->
    lists:reverse(Ps);
decode_params(Data, Ps, Version) ->
    {ParamVal, Data1} = decode_parameter(Data, Version),
    decode_params(Data1, [ParamVal | Ps], Version).




encode_parameter({ParamName, ParamValue}, Ver) ->
    case ParamName of
	q when Ver >= 16#01 ->
	    <<1:1, 16#00:7,
	     (encode_typed_field(Ver,'Q-value', ParamValue))/binary>>;
	charset when Ver >= 16#01 ->
	    <<1:1, 16#01:7,
	     (encode_typed_field(Ver,'Well-known-charset',ParamValue))/binary>>;
	level when Ver >= 16#01 ->
	    <<1:1, 16#02:7,
	     (encode_typed_field(Ver,'Ver-value',ParamValue))/binary>>;

	type when Ver >= ?WSP_12 ->
	    <<1:1, 16#09:7,
	     (encode_typed_field(Ver,'Constrained-encoding',ParamValue))/binary>>;
	type when Ver >= 16#01 ->
	    <<1:1, 16#03:7,
	     (encode_typed_field(Ver,'Integer-value',ParamValue))/binary>>;

	name when Ver >= ?WSP_14 ->
	    <<1:1, 16#17:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	name when Ver >= 16#01 ->
	    <<1:1, 16#05:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

	filename when Ver >= ?WSP_14 ->
	    <<1:1, 16#18:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	filename when Ver >= 16#01 ->
	    <<1:1, 16#06:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

	differences when Ver >= 16#01 ->
	    <<1:1, 16#07:7,
	     (encode_typed_field(Ver,'Field-name',ParamValue))/binary>>;

	padding when Ver >= 16#01 ->
	    <<1:1, 16#08:7,
	     (encode_typed_field(Ver,'Short-integer',ParamValue))/binary>>;


	start when Ver >= ?WSP_14 ->
	    <<1:1, 16#19:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	start when Ver >= ?WSP_12 ->
	    <<1:1, 16#0A:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;


	'start-info' when Ver >= ?WSP_14 ->
	    <<1:1, 16#1A:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	'start-info' when Ver >= ?WSP_12 ->
	    <<1:1, 16#0B:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

        comment when Ver >= ?WSP_14 ->
	    <<1:1, 16#1B:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	comment when Ver >= ?WSP_13 ->
	    <<1:1, 16#0C:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

	domain when Ver >= ?WSP_14 ->
	    <<1:1, 16#1C:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	domain when Ver >= ?WSP_13 ->
	    <<1:1, 16#0D:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

	'max-age' when Ver >= ?WSP_13 ->
	    <<1:1, 16#0E:7,
	     (encode_typed_field(Ver,'Delta-seconds-value',ParamValue))/binary>>;

	path when Ver >= ?WSP_14 ->
	    <<1:1, 16#1D:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	path when Ver >= ?WSP_13 ->
	    <<1:1, 16#0F:7,
	     (encode_typed_field(Ver,'Text-string',ParamValue))/binary>>;

	secure when Ver >= ?WSP_13 ->
	    <<1:1, 16#10:7,
	     (encode_typed_field(Ver,'No-value',ParamValue))/binary>>;
	%% NOTE: "sec" and "mac" are really 1.4 features but used by 1.3 client provisioning
	%"sec" when Ver >= ?WSP_14 ->
	sec when Ver >= ?WSP_13 ->
	    <<1:1, 16#11:7,
	     (encode_typed_field(Ver,'Short-integer',ParamValue))/binary>>;
	%"mac" when Ver >= ?WSP_14 ->
	mac when Ver >= ?WSP_13 ->
	    <<1:1, 16#12:7,
	     (encode_typed_field(Ver,'Text-value',ParamValue))/binary>>;
	'creation-date' when Ver >= ?WSP_14 ->
	    <<1:1, 16#13:7,
	     (encode_typed_field(Ver,'Date-value',ParamValue))/binary>>;
	'modification-date' when Ver >= ?WSP_14 ->
	    <<1:1, 16#14:7,
	     (encode_typed_field(Ver,'Date-value',ParamValue))/binary>>;
	'read-date' when Ver >= ?WSP_14 ->
	    <<1:1, 16#15:7,
	     (encode_typed_field(Ver,'Date-value',ParamValue))/binary>>;
	size when Ver >= ?WSP_14 ->
	    <<1:1, 16#16:7,
	     (encode_typed_field(Ver,'Integer-value',ParamValue))/binary>>;
	_ ->
	    <<(encode_text_string(ParamName))/binary,
	      (encode_text_string(ParamValue))/binary >>
   end.

%% decode_parameter: return {ParameterName, ParamterValue}
decode_parameter(<<1:1,Code:7,Data/binary>>, Version) ->
    case Code of
	16#00 ->
	    {Val,Data1} = decode_typed_field('Q-value', Data, Version),
	    {{ q, Val}, Data1};

	16#01 ->
	    {Val,Data1} = decode_typed_field('Well-known-charset',Data,Version),
	    {{charset, Val}, Data1};

	16#02 ->
	    {Val,Data1} = decode_typed_field('Version-value',Data,Version),
	    {{level, Val}, Data1};

	16#03 ->
	    {Val,Data1} = decode_typed_field('Integer-value', Data,Version),
	    {{type, Val}, Data1};

	16#05 ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{name, Val}, Data1};

	16#06 ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{filename, Val}, Data1};

	16#07 ->
	    {Val,Data1} = decode_typed_field('Field-name', Data,Version),
	    {{differences, Val}, Data1};

	16#08 ->
	    {Val,Data1} = decode_typed_field('Short-integer', Data,Version),
	    {{padding, Val}, Data1};

	16#09 ->
	    {Val,Data1} = decode_typed_field('Constrained-encoding', Data,Version),
	    {{type, Val}, Data1};

	16#0A ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{start, Val}, Data1};

	16#0B ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{'start-info', Val}, Data1};

	16#0C ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{comment, Val}, Data1};

	16#0D ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{domain, Val}, Data1};

	16#0E ->
	    {Val,Data1} = decode_typed_field('Delta-seconds-value', Data,Version),
	    {{'max-age', Val}, Data1};

	16#0F ->
	    {Val,Data1} = decode_typed_field('Text-string', Data,Version),
	    {{path, Val}, Data1};

	16#10 ->
	    {Val,Data1} = decode_typed_field('No-value', Data,Version),
	    {{secure, Val}, Data1};

	16#11 ->
	    {Val,Data1} = decode_typed_field('Short-integer', Data,Version),
	    {{sec, Val}, Data1};

	16#12 ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{mac, Val}, Data1};

	16#13 ->
	    {Val,Data1} = decode_typed_field('Date-value', Data,Version),
	    {{'creation-date', Val}, Data1};

	16#14 ->
	    {Val,Data1} = decode_typed_field('Date-value', Data,Version),
	    {{'modification-date', Val}, Data1};

	16#15 ->
	    {Val,Data1} = decode_typed_field('Date-value', Data,Version),
	    {{'read-date', Val}, Data1};

	16#16 ->
	    {Val,Data1} = decode_typed_field('Integer-value', Data,Version),
	    {{size, Val}, Data1};

	16#17 ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{name, Val}, Data1};

	16#18 ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{filename, Val}, Data1};

	16#19 ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{start, Val}, Data1};

	16#1A ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{'start-info', Val}, Data1};

	16#1B ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{comment, Val}, Data1};

	16#1C ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{domain, Val}, Data1};

	16#1D ->
	    {Val,Data1} = decode_typed_field('Text-value', Data,Version),
	    {{path, Val}, Data1};
	_ ->
	    exit({error, unknown_parameter})
    end;
decode_parameter(Data, _Version) ->
    %% Untyped-parameter: Token-Text Untype-value
    {ParamName,Data1} = d_text_string(Data),
    %% Untype-value: Integer-Value | Text-Value!
    {ParamValue, Data2} = decode_untyped_value(Data1),
    {{ParamName,ParamValue}, Data2}.


encode_typed_field(Ver,Type,Value) ->
    case Type of
	'Well-known-charset' ->
	    MIBenum = encode_charset(Value),
	    encode_integer(MIBenum);

	'Constrained-encoding' ->
	    encode_constrained_media(Value, Ver);

	'Text-string' ->
	    encode_text_string(Value);

	'Text-value' ->
	    encode_text_value(Value);

	'Short-integer' ->
	    ?ENCODE_SHORT(Value);

	'Date-value' ->
	    e_date(Value);

	'Delta-Seconds-value' ->
	    e_delta_seconds(Value);

	'No-value' ->
	    e_no_value(Value);

	_ ->
	    io:format("FIXME: encode_typed_field unsupported type = ~p\n",
		      [Type]),
	    exit({error,badtype})
    end.


decode_typed_field(Type, Data, Version) ->
    case Type of
	'Q-value' ->
	    d_q_value(Data);

	'Well-known-charset' ->
	    {MIBenum, T100} = d_integer_value(Data),
	    {decode_charset(MIBenum), T100};

	'Constrained-encoding' ->
	    {Value, Data1} = scan_header_data(Data),
	    {decode_constrained_media(Value,Version), Data1};

	'Text-string' ->
	    d_text_string(Data);

	'Text-value'  ->
	    d_text_value(Data);

	'Short-integer' ->
	    decode_short_integer(Data);

	'Delta-seconds-value' ->
	    d_integer_value(Data);

	'Date-value' ->
	    {Val, Data1} = decode_long_integer(Data),
	    {d_date(Val), Data1};

	'Field-name' ->
	    d_field_name(Data);

	'No-value' ->
	    d_no_value(Data);

	_ ->
	    io:format("FIXME: unsupported type = ~p\n",[Type]),
	    exit({error,badtype})
    end.


%% Integer-Value | Text-Value
%% return as {Value, Tail}
decode_untyped_value(<<1:1, Short:7, Tail/binary>>) ->
    {Short, Tail};
decode_untyped_value(<<0:3, Len:5, Data/binary>>) when Len =/= 31 ->
    Sz = Len*8,
    <<Long:Sz, Tail/binary>> = Data,
    {Long, Tail};
decode_untyped_value(Data) ->
    d_text_string(Data).


e_field_name(Value, Version) ->
    case normalise_field_name(Value) of
	'Accept' -> <<16#80>>;
	'Accept-Charset' when Version >= ?WSP_13 -> <<16#bb>>;
	'Accept-Charset' -> <<16#81>>;
	'Accept-Encoding' when Version >= ?WSP_13 -> <<16#bc>>;
	'Accept-Encoding' -> <<16#82>>;
	'Accept-Language' -> <<16#83>>;
	'Accept-Ranges' -> <<16#84>>;
	'Age' -> <<16#85>>;
	'Allow' -> <<16#86>>;
	'Authorization' -> <<16#87>>;
	'Cache-Control' when Version >= ?WSP_14 -> <<16#c7>>;
	'Cache-Control' when Version >= ?WSP_13 -> <<16#bd>>;
	'Cache-Control' -> <<16#88>>;
	'Connection' -> <<16#89>>;
	'Content-Base' -> <<16#8a>>;
	'Content-Encoding' -> <<16#8b>>;
	'Content-Language' -> <<16#8c>>;
	'Content-Length' -> <<16#8d>>;
	'Content-Location' -> <<16#8e>>;
	'Content-Md5' -> <<16#8f>>;
	'Content-Range' when Version >= ?WSP_13 -> <<16#be>>;
	'Content-Range' -> <<16#90>>;
	'Content-Type' -> <<16#91>>;
	'Date' -> <<16#92>>;
	'Etag' -> <<16#93>>;
	'Expires' -> <<16#94>>;
	'From' -> <<16#95>>;
	'Host' -> <<16#96>>;
	'If-Modified-Since' -> <<16#97>>;
	'If-Match' -> <<16#98>>;
	'If-None-Match' -> <<16#99>>;
	'If-Range' -> <<16#9a>>;
	'If-Unmodified-Since' -> <<16#9b>>;
	'Location' -> <<16#9c>>;
	'Last-Modified' -> <<16#9d>>;
	'Max-Forwards' -> <<16#9e>>;
	'Pragma' -> <<16#9f>>;
	'Proxy-Authenticate' -> <<16#a0>>;
	'Proxy-Authorization' -> <<16#a1>>;
	'Public' -> <<16#a2>>;
	'Range' -> <<16#a3>>;
	'Referer' -> <<16#a4>>;
	'Retry-After' -> <<16#a5>>;
	'Server' -> <<16#a6>>;
	'Transfer-Encoding' -> <<16#a7>>;
	'Upgrade' -> <<16#a8>>;
	'User-Agent' -> <<16#a9>>;
	'Vary' -> <<16#aa>>;
	'Via' -> <<16#ab>>;
	'Warning' -> <<16#ac>>;
	'Www-Authenticate' -> <<16#ad>>;
	'Content-Disposition' when Version >= ?WSP_14 -> <<16#c5>>;
	'Content-Disposition' -> <<16#ae>>;
	%% VERSION > 1.1
	'X-Wap-Application-Id' when Version >= ?WSP_12 -> <<16#af>>;
	'X-Wap-Content-Uri' when Version >= ?WSP_12 -> <<16#b0>>;
	'X-Wap-Initiator-Uri' when Version >= ?WSP_12 -> <<16#b1>>;
	'Accept-Application' when Version >= ?WSP_12 -> <<16#b2>>;
	'Bearer-Indication' when Version >= ?WSP_12 -> <<16#b3>>;
	'Push-Flag' when Version >= ?WSP_12 -> <<16#b4>>;
	'Profile' when Version >= ?WSP_12 -> <<16#b5>>;
	'Profile-Diff' when Version >= ?WSP_12 -> <<16#b6>>;
	'Profile-Warning' when Version >= ?WSP_12 -> <<16#b7>>;
	'Expect' when Version >= ?WSP_15 -> <<16#c8>>;
	'Expect' when Version >= ?WSP_13 -> <<16#b8>>;
	'Te' when Version >= ?WSP_13 -> <<16#b9>>;
	'Trailer' when Version >= ?WSP_13 -> <<16#ba>>;
	'X-Wap-Tod' when Version >= ?WSP_13 -> <<16#bf>>;
	'Content-Id' when Version >= ?WSP_13 -> <<16#c0>>;
	'Set-Cookie' when Version >= ?WSP_13 -> <<16#c1>>;
	'Cookie' when Version >= ?WSP_13 -> <<16#c2>>;
	'Encoding-Version' when Version >= ?WSP_13 -> <<16#c3>>;
	'Profile-Warning' when Version >= ?WSP_14 -> <<16#c4>>;
	'X-Wap-Security' when Version >= ?WSP_14 -> <<16#c6>>;
	'X-Wap-Loc-Invocation' when Version >= ?WSP_15 -> <<16#c9>>;
	'X-Wap-Loc-Delivery' when Version >= ?WSP_15 -> <<16#ca>>;
	Field -> encode_text_string(atom_to_list(Field))
    end.


%%
%% decode and normalise on form list_to_atom("Ulll-Ulll-Ull")
%%
normalise_field_name(Cs) when atom(Cs) ->
    Cs;
normalise_field_name(Cs) ->
    list_to_atom(normalise_fieldU(Cs)).

normalise_fieldU([C|Cs]) when C >= $a, C =< $z ->
    [(C-$a)+$A | normalise_fieldL(Cs)];
normalise_fieldU([C|Cs]) -> [ C | normalise_fieldL(Cs)];
normalise_fieldU([]) -> [].

normalise_fieldL([C|Cs]) when C >= $A, C =< $Z ->
    [(C-$A)+$a | normalise_fieldL(Cs)];
normalise_fieldL([$-|Cs]) ->  [$- | normalise_fieldU(Cs)];
normalise_fieldL([C|Cs]) ->   [C | normalise_fieldL(Cs)];
normalise_fieldL([]) ->  [].


tolower([C|Cs]) when C >= $A, C =< $Z ->
    [(C-$A)+$a | tolower(Cs)];
tolower([C|Cs]) ->  [C|tolower(Cs)];
tolower([]) -> [].

trim(Cs) ->
    lists:reverse(trim1(lists:reverse(trim1(Cs)))).

trim1([$\s|Cs]) -> trim1(Cs);
trim1([$\t|Cs]) -> trim1(Cs);
trim1([$\r|Cs]) -> trim1(Cs);
trim1([$\n|Cs]) -> trim1(Cs);
trim1(Cs) -> Cs.


d_field_name(Data) ->
    case scan_header_data(Data) of
	{Code, Data1} when integer(Code) ->
	    {lookup_field_name(Code), Data1};
	{TmpField,Data1} when list(TmpField) ->
	    {normalise_field_name(TmpField), Data1}
    end.

d_no_value(<<0, Data/binary>>) ->
    {no_value, Data}.

e_no_value(_) ->
    <<0>>.


lookup_field_name(Code) ->
    case Code of
%%% Version 1.1
	16#00 -> 'Accept';
	16#01 -> 'Accept-Charset';
	16#02 -> 'Accept-Encoding';
	16#03 -> 'Accept-Language';
	16#04 -> 'Accept-Ranges';
	16#05 -> 'Age';
	16#06 -> 'Allow';
	16#07 -> 'Authorization';
	16#08 -> 'Cache-Control';
	16#09 -> 'Connection';
	16#0a -> 'Content-Base';
	16#0b -> 'Content-Encoding';
	16#0c -> 'Content-Language';
	16#0d -> 'Content-Length';
	16#0e -> 'Content-Location';
	16#0f -> 'Content-Md5';
	16#10 -> 'Content-Range';
	16#11 -> 'Content-Type';
	16#12 -> 'Date';
	16#13 -> 'Etag';
	16#14 -> 'Expires';
	16#15 -> 'From';
	16#16 -> 'Host';
	16#17 -> 'If-Modified-Since';
	16#18 -> 'If-Match';
	16#19 -> 'If-None-Match';
	16#1a -> 'If-Range';
	16#1b -> 'If-Unmodified-Since';
	16#1c -> 'Location';
	16#1d -> 'Last-Modified';
	16#1e -> 'Max-Forwards';
	16#1f -> 'Pragma';
	16#20 -> 'Proxy-Authenticate';
	16#21 -> 'Proxy-Authorization';
	16#22 -> 'Public';
	16#23 -> 'Range';
	16#24 -> 'Referer';
	16#25 -> 'Retry-After';
	16#26 -> 'Server';
	16#27 -> 'Transfer-Encoding';
	16#28 -> 'Upgrade';
	16#29 -> 'User-Agent';
	16#2a -> 'Vary';
	16#2b -> 'Via';
	16#2c -> 'Warning';
	16#2d -> 'Www-Authenticate';
	16#2e -> 'Content-Disposition';
%%% Version 1.2
	16#2f -> 'X-Wap-Application-Id';
	16#30 -> 'X-Wap-Content-Uri';
	16#31 -> 'X-Wap-Initiator-Uri';
	16#32 -> 'Accept-Application';
	16#33 -> 'Bearer-Indication';
	16#34 -> 'Push-Flag';
	16#35 -> 'Profile';
	16#36 -> 'Profile-Diff';
	16#37 -> 'Profile-Warning';
%%% Version 1.3
	16#38 -> 'Expect';
	16#39 -> 'Te';
	16#3a -> 'Trailer';
	16#3b -> 'Accept-Charset';
	16#3c -> 'Accept-Encoding';
	16#3d -> 'Cache-Control';
	16#3e -> 'Content-Range';
	16#3f -> 'X-Wap-Tod';
	16#40 -> 'Content-Id';
	16#41 -> 'Set-Cookie';
	16#42 -> 'Cookie';
	16#43 -> 'Encoding-Version';
%%% Version 1.4
	16#44 -> 'Profile-Warning';
	16#45 -> 'Content-Disposition';
	16#46 -> 'X-Wap-Security';
	16#47 -> 'Cache-Control';
%%% Version 1.5
	16#48 -> 'Expect';
	16#49 -> 'X-Wap-Loc-Invocation';
	16#4a -> 'X-Wap-Loc-Delivery';
%% Unknown
	_ ->
	    list_to_atom("X-Unknown-"++erlang:integer_to_list(Code, 16))
    end.


encode_charset(Charset) ->
    %% FIXME: we should really resolve aliases as well
    %% charset:from_aliases(Charset)
    case charset:from_mime_name(Charset) of
	0 -> exit({error, unknown_charset});
	MIBenum	-> MIBenum
    end.

encode_language(Language) ->
    Code = encode_lang(tolower(Language)),
    <<Code>>.



decode_charset(MIBenum) ->
    case charset:to_mime_name(MIBenum) of
	undefined ->
	    exit({error, unknown_charset});
	Preferred ->
	    Preferred
    end.

%% ISO 639 Language Assignments, Appendix A, Table 41, Page 102-103
decode_lang(Code) ->
    case lookup_language(Code) of
	[L|_] -> atom_to_list(L);
	[] -> ""
    end.


lookup_language(Code) ->
    case Code of
	 16#01 -> ['aa','afar'];
	 16#02 -> ['ab','abkhazian'];
	 16#03 -> ['af','afrikans'];
	 16#04 -> ['am','amharic'];
	 16#05 -> ['ar','arabic'];
	 16#06 -> ['as','assamese'];
	 16#07 -> ['ay','aymara'];
	 16#08 -> ['az','azerbaijani'];
	 16#09 -> ['ba','bashkir'];
	 16#0a -> ['be','byelorussian'];
	 16#0b -> ['bg','bulgarian'];
	 16#0c -> ['bh','bihari'];
	 16#0d -> ['bi','bislama'];
	 16#0e -> ['bn','bangla','bengali'];
	 16#0f -> ['bo','tibetan'];
	 16#10 -> ['br','breton'];
	 16#11 -> ['ca','catalan'];
	 16#12 -> ['co','corsican'];
	 16#13 -> ['cs','czech'];
	 16#14 -> ['cy','welsh'];
	 16#15 -> ['da','danish'];
	 16#16 -> ['de','german'];
	 16#17 -> ['dz','bhutani'];
	 16#18 -> ['el','greek'];
	 16#19 -> ['en','english'];
	 16#1a -> ['eo','esperanto'];
	 16#1b -> ['es','spanish'];
	 16#1c -> ['et','estonian'];
	 16#1d -> ['eu','basque'];
	 16#1e -> ['fa','persian'];
	 16#1f -> ['fi','finnish'];
	 16#20 -> ['fj','fiji'];
	 16#82 -> ['fo','faeroese'];
	 16#22 -> ['fr','french'];
	 16#83 -> ['fy','frisian'];
	 16#24 -> ['ga','irish'];
	 16#25 -> ['gd','scots-gaelic'];
	 16#26 -> ['gl','galician'];
	 16#27 -> ['gn','guarani'];
	 16#28 -> ['gu','gujarati'];
	 16#29 -> ['ha','hausa'];
	 16#2a -> ['he','hebrew'];
	 16#2b -> ['hi','hindi'];
	 16#2c -> ['hr','croatian'];
	 16#2d -> ['hu','hungarian'];
	 16#2e -> ['hy','armenian'];
	 16#84 -> ['ia','interlingua'];
	 16#30 -> ['id','indonesian'];
	 16#86 -> ['ie','interlingue'];
	 16#87 -> ['ik','inupiak'];
	 16#33 -> ['is','icelandic'];
	 16#34 -> ['it','italian'];
	 16#89 -> ['iu','inuktitut'];
	 16#36 -> ['ja','japanese'];
	 16#37 -> ['jw','javanese'];
	 16#38 -> ['ka','georgian'];
	 16#39 -> ['kk','kazakh'];
	 16#8a -> ['kl','greenlandic'];
	 16#3b -> ['km','cambodian'];
	 16#3c -> ['kn','kannada'];
	 16#3d -> ['ko','korean'];
	 16#3e -> ['ks','kashmiri'];
	 16#3f -> ['ku','kurdish'];
	 16#40 -> ['ky','kirghiz'];
	 16#8b -> ['la','latin'];
	 16#42 -> ['ln','lingala'];
	 16#43 -> ['lo','laothian'];
	 16#44 -> ['lt','lithuanian'];
	 16#45 -> ['lv','lettish','latvian'];
	 16#46 -> ['mg','malagese'];
	 16#47 -> ['mi','maori'];
	 16#48 -> ['mk','macedonian'];
	 16#49 -> ['ml','malayalam'];
	 16#4a -> ['mn','mongolian'];
	 16#4b -> ['mo','moldavian'];
	 16#4c -> ['mr','marathi'];
	 16#4d -> ['ms','malay'];
	 16#4e -> ['mt','maltese'];
	 16#4f -> ['my','burmese'];
	 16#81 -> ['na','nauru'];
	 16#51 -> ['ne','nepali'];
	 16#52 -> ['nl','dutch'];
	 16#53 -> ['no','norwegian'];
	 16#54 -> ['oc','occitan'];
	 16#55 -> ['om','oromo'];
	 16#56 -> ['or','oriya'];
	 16#57 -> ['pa','punjabi'];
	 16#58 -> ['po','polish'];
	 16#59 -> ['ps','pushto','pashto'];
	 16#5a -> ['pt','portugese'];
	 16#5b -> ['qu','quechua'];
	 16#8c -> ['rm','rhaeto-romance'];
	 16#5d -> ['rn','kirundi'];
	 16#5e -> ['ro','romanian'];
	 16#5f -> ['ru','russian'];
	 16#60 -> ['rw','kinyarwanda'];
	 16#61 -> ['sa','sanskrit'];
	 16#62 -> ['sd','sindhi'];
	 16#63 -> ['sg','sangho'];
	 16#64 -> ['sh','serbo-croatian'];
	 16#65 -> ['si','sinhalese'];
	 16#66 -> ['sk','slovak'];
	 16#67 -> ['sl','slovenian'];
	 16#68 -> ['sm','samoan'];
	 16#69 -> ['sn','shona'];
	 16#6a -> ['so','somali'];
	 16#6b -> ['sq','albanian'];
	 16#6c -> ['sr','serbian'];
	 16#6d -> ['ss','siswati'];
	 16#6e -> ['st','seshoto'];
	 16#6f -> ['su','sundanese'];
	 16#70 -> ['sv','swedish'];
	 16#71 -> ['sw','swahili'];
	 16#72 -> ['ta','tamil'];
	 16#73 -> ['te','telugu'];
	 16#74 -> ['tg','tajik'];
	 16#75 -> ['th','thai'];
	 16#76 -> ['ti','tigrinya'];
	 16#77 -> ['tk','turkmen'];
	 16#78 -> ['tl','tagalog'];
	 16#79 -> ['tn','setswana'];
	 16#7a -> ['to','tonga'];
	 16#7b -> ['tr','turkish'];
	 16#7c -> ['ts','tsonga'];
	 16#7d -> ['tt','tatar'];
	 16#7e -> ['tw','twi'];
	 16#7f -> ['ug','uighur'];
	 16#50 -> ['uk','ukrainian'];
	 16#21 -> ['ur','urdu'];
	 16#23 -> ['uz','uzbek'];
	 16#2f -> ['vi','vietnamese'];
	 16#85 -> ['vo','volapuk'];
	 16#31 -> ['wo','wolof'];
	 16#32 -> ['xh','xhosa'];
	 16#88 -> ['yi','yiddish'];
	 16#35 -> ['yo','yoruba'];
	 16#3a -> ['za','zhuang'];
	 16#41 -> ['zh','chinese'];
	16#5c -> ['zu','zulu'];
	_ -> []
    end.

encode_lang(Language) ->
    case tolower(Language) of
	"aa" -> 16#01;
	"afar" -> 16#01;
	"ab" -> 16#02;
	"abkhazian" -> 16#02;
	"af" -> 16#03;
	"afrikans" -> 16#03;
	"am" -> 16#04;
	"amharic" -> 16#04;
	"ar" -> 16#05;
	"arabic" -> 16#05;
	"as" -> 16#06;
	"assamese" -> 16#06;
	"ay" -> 16#07;
	"aymara" -> 16#07;
	"az" -> 16#08;
	"azerbaijani" -> 16#08;
	"ba" -> 16#09;
	"bashkir" -> 16#09;
	"be" -> 16#0a;
	"byelorussian" -> 16#0a;
	"bg" -> 16#0b;
	"bulgarian" -> 16#0b;
	"bh" -> 16#0c;
	"bihari" -> 16#0c;
	"bi" -> 16#0d;
	"bislama" -> 16#0d;
	"bn" -> 16#0e;
	"bangla" -> 16#0e;
	"bengali" -> 16#0e;
	"bo" -> 16#0f;
	"tibetan" -> 16#0f;
	"br" -> 16#10;
	"breton" -> 16#10;
	"ca" -> 16#11;
	"catalan" -> 16#11;
	"co" -> 16#12;
	"corsican" -> 16#12;
	"cs" -> 16#13;
	"czech" -> 16#13;
	"cy" -> 16#14;
	"welsh" -> 16#14;
	"da" -> 16#15;
	"danish" -> 16#15;
	"de" -> 16#16;
	"german" -> 16#16;
	"dz" -> 16#17;
	"bhutani" -> 16#17;
	"el" -> 16#18;
	"greek" -> 16#18;
	"en" -> 16#19;
	"english" -> 16#19;
	"eo" -> 16#1a;
	"esperanto" -> 16#1a;
	"es" -> 16#1b;
	"spanish" -> 16#1b;
	"et" -> 16#1c;
	"estonian" -> 16#1c;
	"eu" -> 16#1d;
	"basque" -> 16#1d;
	"fa" -> 16#1e;
	"persian" -> 16#1e;
	"fi" -> 16#1f;
	"finnish" -> 16#1f;
	"fj" -> 16#20;
	"fiji" -> 16#20;
	"fo" -> 16#82;
	"faeroese" -> 16#82;
	"fr" -> 16#22;
	"french" -> 16#22;
	"fy" -> 16#83;
	"frisian" -> 16#83;
	"ga" -> 16#24;
	"irish" -> 16#24;
	"gd" -> 16#25;
	"scots-gaelic" -> 16#25;
	"gl" -> 16#26;
	"galician" -> 16#26;
	"gn" -> 16#27;
	"guarani" -> 16#27;
	"gu" -> 16#28;
	"gujarati" -> 16#28;
	"ha" -> 16#29;
	"hausa" -> 16#29;
	"he" -> 16#2a;
	"hebrew" -> 16#2a;
	"hi" -> 16#2b;
	"hindi" -> 16#2b;
	"hr" -> 16#2c;
	"croatian" -> 16#2c;
	"hu" -> 16#2d;
	"hungarian" -> 16#2d;
	"hy" -> 16#2e;
	"armenian" -> 16#2e;
	"ia" -> 16#84;
	"interlingua" -> 16#84;
	"id" -> 16#30;
	"indonesian" -> 16#30;
	"ie" -> 16#86;
	"interlingue" -> 16#86;
	"ik" -> 16#87;
	"inupiak" -> 16#87;
	"is" -> 16#33;
	"icelandic" -> 16#33;
	"it" -> 16#34;
	"italian" -> 16#34;
	"iu" -> 16#89;
	"inuktitut" -> 16#89;
	"ja" -> 16#36;
	"japanese" -> 16#36;
	"jw" -> 16#37;
	"javanese" -> 16#37;
	"ka" -> 16#38;
	"georgian" -> 16#38;
	"kk" -> 16#39;
	"kazakh" -> 16#39;
	"kl" -> 16#8a;
	"greenlandic" -> 16#8a;
	"km" -> 16#3b;
	"cambodian" -> 16#3b;
	"kn" -> 16#3c;
	"kannada" -> 16#3c;
	"ko" -> 16#3d;
	"korean" -> 16#3d;
	"ks" -> 16#3e;
	"kashmiri" -> 16#3e;
	"ku" -> 16#3f;
	"kurdish" -> 16#3f;
	"ky" -> 16#40;
	"kirghiz" -> 16#40;
	"la" -> 16#8b;
	"latin" -> 16#8b;
	"ln" -> 16#42;
	"lingala" -> 16#42;
	"lo" -> 16#43;
	"laothian" -> 16#43;
	"lt" -> 16#44;
	"lithuanian" -> 16#44;
	"lv" -> 16#45;
	"lettish" -> 16#45;
	"latvian" -> 16#45;
	"mg" -> 16#46;
	"malagese" -> 16#46;
	"mi" -> 16#47;
	"maori" -> 16#47;
	"mk" -> 16#48;
	"macedonian" -> 16#48;
	"ml" -> 16#49;
	"malayalam" -> 16#49;
	"mn" -> 16#4a;
	"mongolian" -> 16#4a;
	"mo" -> 16#4b;
	"moldavian" -> 16#4b;
	"mr" -> 16#4c;
	"marathi" -> 16#4c;
	"ms" -> 16#4d;
	"malay" -> 16#4d;
	"mt" -> 16#4e;
	"maltese" -> 16#4e;
	"my" -> 16#4f;
	"burmese" -> 16#4f;
	"na" -> 16#81;
	"nauru" -> 16#81;
	"ne" -> 16#51;
	"nepali" -> 16#51;
	"nl" -> 16#52;
	"dutch" -> 16#52;
	"no" -> 16#53;
	"norwegian" -> 16#53;
	"oc" -> 16#54;
	"occitan" -> 16#54;
	"om" -> 16#55;
	"oromo" -> 16#55;
	"or" -> 16#56;
	"oriya" -> 16#56;
	"pa" -> 16#57;
	"punjabi" -> 16#57;
	"po" -> 16#58;
	"polish" -> 16#58;
	"ps" -> 16#59;
	"pushto" -> 16#59;
	"pt" -> 16#5a;
	"portugese" -> 16#5a;
	"qu" -> 16#5b;
	"quechua" -> 16#5b;
	"rm" -> 16#8c;
	"rhaeto-romance" -> 16#8c;
	"rn" -> 16#5d;
	"kirundi" -> 16#5d;
	"ro" -> 16#5e;
	"romanian" -> 16#5e;
	"ru" -> 16#5f;
	"russian" -> 16#5f;
	"rw" -> 16#60;
	"kinyarwanda" -> 16#60;
	"sa" -> 16#61;
	"sanskrit" -> 16#61;
	"sd" -> 16#62;
	"sindhi" -> 16#62;
	"sg" -> 16#63;
	"sangho" -> 16#63;
	"sh" -> 16#64;
	"serbo-croatian" -> 16#64;
	"si" -> 16#65;
	"sinhalese" -> 16#65;
	"sk" -> 16#66;
	"slovak" -> 16#66;
	"sl" -> 16#67;
	"slovenian" -> 16#67;
	"sm" -> 16#68;
	"samoan" -> 16#68;
	"sn" -> 16#69;
	"shona" -> 16#69;
	"so" -> 16#6a;
	"somali" -> 16#6a;
	"sq" -> 16#6b;
	"albanian" -> 16#6b;
	"sr" -> 16#6c;
	"serbian" -> 16#6c;
	"ss" -> 16#6d;
	"siswati" -> 16#6d;
	"st" -> 16#6e;
	"seshoto" -> 16#6e;
	"su" -> 16#6f;
	"sundanese" -> 16#6f;
	"sv" -> 16#70;
	"swedish" -> 16#70;
	"sw" -> 16#71;
	"swahili" -> 16#71;
	"ta" -> 16#72;
	"tamil" -> 16#72;
	"te" -> 16#73;
	"telugu" -> 16#73;
	"tg" -> 16#74;
	"tajik" -> 16#74;
	"th" -> 16#75;
	"thai" -> 16#75;
	"ti" -> 16#76;
	"tigrinya" -> 16#76;
	"tk" -> 16#77;
	"turkmen" -> 16#77;
	"tl" -> 16#78;
	"tagalog" -> 16#78;
	"tn" -> 16#79;
	"setswana" -> 16#79;
	"to" -> 16#7a;
	"tonga" -> 16#7a;
	"tr" -> 16#7b;
	"turkish" -> 16#7b;
	"ts" -> 16#7c;
	"tsonga" -> 16#7c;
	"tt" -> 16#7d;
	"tatar" -> 16#7d;
	"tw" -> 16#7e;
	"twi" -> 16#7e;
	"ug" -> 16#7f;
	"uighur" -> 16#7f;
	"uk" -> 16#50;
	"ukrainian" -> 16#50;
	"ur" -> 16#21;
	"urdu" -> 16#21;
	"uz" -> 16#23;
	"uzbek" -> 16#23;
	"vi" -> 16#2f;
	"vietnamese" -> 16#2f;
	"vo" -> 16#85;
	"volapuk" -> 16#85;
	"wo" -> 16#31;
	"wolof" -> 16#31;
	"xh" -> 16#32;
	"xhosa" -> 16#32;
	"yi" -> 16#88;
	"yiddish" -> 16#88;
	"yo" -> 16#35;
	"yoruba" -> 16#35;
	"za" -> 16#3a;
	"zhuang" -> 16#3a;
	"zh" -> 16#41;
	"chinese" -> 16#41;
	"zu" -> 16#5c;
	"zulu" -> 16#5c
    end.


%% Push Application ID Assignments
%%
%% Assingment are found at http://www.wapforum.org/wina/push-app-id.htm
%%
decode_push_application({short,Data}) ->
    decode_push_application(d_long(Data));

decode_push_application(Code) when integer(Code) ->
    case Code of
	16#00 ->    "x-wap-application:*";
	16#01 ->    "x-wap-application:push.sia";
	16#02 ->    "x-wap-application:wml.ua";
	16#03 ->    "x-wap-application:wta.ua";
	16#04 ->    "x-wap-application:mms.ua";
	16#05 ->    "x-wap-application:push.syncml";
	16#06 ->    "x-wap-application:loc.ua";
	16#07 ->    "x-wap-application:syncml.dm";
	16#08 ->    "x-wap-application:drm.ua";
	16#09 ->    "x-wap-application:emn.ua";
	16#0A ->    "x-wap-application:wv.ua";
	16#8000 ->  "x-wap-microsoft:localcontent.ua";
	16#8001 ->  "x-wap-microsoft:IMclient.ua";
	16#8002 ->  "x-wap-docomo:imode.mail.ua";
	16#8003 ->  "x-wap-docomo:imode.mr.ua";
	16#8004 ->  "x-wap-docomo:imode.mf.ua";
	16#8005 ->  "x-motorola:location.ua";
	16#8006 ->  "x-motorola:now.ua";
	16#8007 ->  "x-motorola:otaprov.ua";
	16#8008 ->  "x-motorola:browser.ua";
	16#8009 ->  "x-motorola:splash.ua";
	16#800B ->  "x-wap-nai:mvsw.command";
	16#8010 ->  "x-wap-openwave:iota.ua"
    end;
decode_push_application(App) when list(App) ->
    App.



encode_push_application(App) ->
    case App of
	"x-wap-application:*" -> ?ENCODE_SHORT(16#00);
	"x-wap-application:push.sia" -> ?ENCODE_SHORT(16#01);
	"x-wap-application:wml.ua" -> ?ENCODE_SHORT(16#02);
	"x-wap-application:wta.ua" -> ?ENCODE_SHORT(16#03);
	"x-wap-application:mms.ua" -> ?ENCODE_SHORT(16#04);
	"x-wap-application:push.syncml" -> ?ENCODE_SHORT(16#05);
	"x-wap-application:loc.ua" -> ?ENCODE_SHORT(16#06);
	"x-wap-application:syncml.dm" -> ?ENCODE_SHORT(16#07);
	"x-wap-application:drm.ua" -> ?ENCODE_SHORT(16#08);
	"x-wap-application:emn.ua" -> ?ENCODE_SHORT(16#09);
	"x-wap-application:wv.ua" -> ?ENCODE_SHORT(16#0A);
	"x-wap-microsoft:localcontent.ua" -> encode_integer(16#8000);
	"x-wap-microsoft:IMclient.ua" -> encode_integer(16#8001);
	"x-wap-docomo:imode.mail.ua" -> encode_integer(16#8002);
	"x-wap-docomo:imode.mr.ua" -> encode_integer(16#8003);
	"x-wap-docomo:imode.mf.ua" -> encode_integer(16#8004);
	"x-motorola:location.ua" -> encode_integer(16#8005);
	"x-motorola:now.ua" -> encode_integer(16#8006);
	"x-motorola:otaprov.ua" -> encode_integer(16#8007);
	"x-motorola:browser.ua" -> encode_integer(16#8008);
	"x-motorola:splash.ua" -> encode_integer(16#8009);
	"x-wap-nai:mvsw.command" -> encode_integer(16#800B);
	"x-wap-openwave:iota.ua" -> encode_integer(16#8010);
	_ -> encode_uri_value(App)
    end.




%% WSP 8.5 Multipart handling

encode_multipart(Entries) ->
    encode_multipart(Entries, ?WSP_DEFAULT_VERSION).

encode_multipart([], _Version) ->
    <<>>;
encode_multipart(Entries, Version) ->
    EncEntries = encode_multipart_entries(Entries, Version),
    <<(e_uintvar(length(Entries)))/binary, EncEntries/binary >>.

encode_multipart_entries(Entries, Version) ->
    encode_multipart_entries(Entries, Version, []).

encode_multipart_entries([], _Version, Acc) ->
    list_to_binary(lists:reverse(Acc));
encode_multipart_entries([Entry|T], Version, Acc) ->
    EncEntry = encode_multipart_entry(Entry, Version),
    encode_multipart_entries(T, Version, [EncEntry | Acc]).

encode_multipart_entry(Entry, Version) ->
    #wsp_multipart_entry { content_type = ContentType,
			   headers = Headers,
			   data = Data } = Entry,
    EncContentType = encode_content_type(ContentType,Version),
    EncHeaders = encode_headers(Headers, Version),
    EncHeadersLength = e_uintvar(size(EncContentType)+size(EncHeaders)),
    DataLen = e_uintvar(size(Data)),
    <<EncHeadersLength/binary,
     DataLen/binary,
     EncContentType/binary,
     EncHeaders/binary,
     Data/binary>>.


decode_multipart(Data) ->
    decode_multipart(Data, ?WSP_DEFAULT_VERSION).

decode_multipart(<<>>, _Version) ->
    {[], <<>>};
decode_multipart(Data, Version) ->
    {Entries, Data1} = d_uintvar(Data),
    decode_multipart_entries(Entries, Data1, Version).

decode_multipart_entries(Entries, Data, Version) ->
    decode_multipart_entries(Entries, Data, Version, []).

decode_multipart_entries(0, Data, _Version, Acc) ->
    {lists:reverse(Acc), Data};
decode_multipart_entries(Entries, Data, Version, Acc) ->
    {MultiPartEntry, Data1} = decode_multipart_entry(Data,Version),
    decode_multipart_entries(Entries-1, Data1, Version, [MultiPartEntry|Acc]).

decode_multipart_entry(Data, Version) ->
    {HeadersLen, Data1} = d_uintvar(Data),
    {DataLen, Data2} = d_uintvar(Data1),
    {FieldData,Data3} = scan_header_data(Data2),
    ContentType = decode_content_type(FieldData, Version),
    BinHeadersLen = (HeadersLen-(size(Data2)-size(Data3))),
    <<BinHeaders:BinHeadersLen/binary,Data4/binary>> = Data3,
    Headers = decode_headers(BinHeaders, Version),
    <<ValueData:DataLen/binary, Data5/binary>> = Data4,
    {#wsp_multipart_entry{content_type=ContentType,
			  headers=Headers,
			  data=ValueData},Data5}.


parse_credentials(Field, Value) ->
    %% FIXME
    ?WH(Field, Value, []).

format_credentials("basic", [User,Password]) ->
    ["Basic ", base64:encode(User++":"++Password)];
format_credentials(Scheme, Params) ->
    [Scheme, format_params(Params)].

encode_credentials("basic", [User,Password], _Version) ->
    e_value(?ENCODE_SHORT(0),
	    encode_text_string(User),
	    encode_text_string(Password));
encode_credentials(Scheme, Params, Version) ->
    e_value(encode_text_string(Scheme), encode_params(Params, Version)).

decode_credentials(Field, Data, Version) ->
    case scan_header_data(Data) of
	{0, Data0} ->
	    {User,Data1} = d_text_string(Data0),
	    {Password,_Data2} = d_text_string(Data1),
	    ?WH(Field, "basic", [User,Password]);
	{Scheme, Data0} when list(Scheme) ->
	    Params = decode_params(Data0, Version),
	    ?WH(Field, Scheme, Params)
    end.

%%
%% Challenge: Basic Realm-value | Auth-Scheme Realm *Auth-Params
%%

parse_challenge(Field, Value) ->
    %% FIXME
    ?WH(Field, Value, []).

format_challenge({"basic",Realm}, []) ->
    ["Basic ", Realm];
format_challenge({Scheme,Realm}, Params) ->
    [Scheme," ",Realm, format_params(Params)].

encode_challenge({"basic",Realm}, [], _Version) ->
    e_value(?ENCODE_SHORT(0),
	    encode_text_string(Realm));
encode_challenge({Scheme,Realm}, Params, Version) ->
    e_value(encode_text_string(Scheme),
	    encode_text_string(Realm),
	    encode_params(Params, Version)).

decode_challenge(Field, Data, Version) ->
    case scan_header_data(Data) of
	{0, Data0} ->
	    {Realm,_} = d_text_string(Data0),
	    ?WH(Field, {"basic", Realm}, []);
	{Scheme, Data0} when list(Scheme) ->
	    {Realm,_} = d_text_string(Data0),
	    Params = decode_params(Data0, Version),
	    ?WH(Field, {Scheme,Realm}, Params)
    end.


parse_well_known_method(Value) ->
    case Value of
	"GET" ->  'GET';
	"OPTIONS" -> 'OPTIONS';
	"HEAD" -> 'HEAD';
	"DELETE" -> 'DELETE';
	"TRACE" -> 'TRACE';
	"POST" -> 'POST';
	"PUT" -> 'PUT'
    end.

encode_well_known_method(Value, _Version) ->
    case Value of
	'GET'     -> ?ENCODE_SHORT(16#40);
	'OPTIONS' -> ?ENCODE_SHORT(16#41);
	'HEAD'    -> ?ENCODE_SHORT(16#42);
	'DELETE'  -> ?ENCODE_SHORT(16#43);
	'TRACE'   -> ?ENCODE_SHORT(16#44);
	'POST'    -> ?ENCODE_SHORT(16#60);
	'PUT'     -> ?ENCODE_SHORT(16#61)
    end.

decode_well_known_method(Value, _Version) ->
    case Value of
	16#40 -> 'GET';
	16#41 -> 'OPTIONS';
	16#42 -> 'HEAD';
	16#43 -> 'DELETE';
	16#44 -> 'TRACE';
	16#60 -> 'POST';
	16#61 -> 'PUT'
    end.



%%
%% WSP Table 36. Status Code Assignments
%%

encode_status_code(Status) ->
    case Status of
	100 -> 16#10; %% 'Continue'
	101 -> 16#11; %% 'Switching Protocols'
	200 -> 16#20; %% 'OK, Success'
	201 -> 16#21; %% 'Created'
	202 -> 16#22; %% 'Accepted'
	203 -> 16#23; %% 'Non-Authoritative Information'
	204 -> 16#24; %% 'No Content'
	205 -> 16#25; %% 'Reset Content'
	206 -> 16#26; %% 'Partial Content'
	300 -> 16#30; %% 'Multiple Choices'
	301 -> 16#31; %% 'Moved Permanently'
	302 -> 16#32; %% 'Moved temporarily'
	303 -> 16#33; %% 'See Other'
	304 -> 16#34; %% 'Not modified'
	305 -> 16#35; %% 'Use Proxy'
	306 -> 16#36; %% '(reserved)'
	307 -> 16#37; %% 'Temporary Redirect'
	400 -> 16#40; %% 'Bad Request - server could not understand request'
	401 -> 16#41; %% 'Unauthorized'
	402 -> 16#42; %% 'Payment required'
	403 -> 16#43; %% 'Forbidden operation is understood but refused'
	404 -> 16#44; %% 'Not Found'
	405 -> 16#45; %% 'Method not allowed'
	406 -> 16#46; %% 'Not Acceptable'
	407 -> 16#47; %% 'Proxy Authentication required'
	408 -> 16#48; %% 'Request Timeout'
	409 -> 16#49; %% 'Conflict'
	410 -> 16#4A; %% 'Gone'
	411 -> 16#4B; %% 'Length Required'
	412 -> 16#4C; %% 'Precondition failed'
	413 -> 16#4D; %% 'Request entity too large'
	414 -> 16#4E; %% 'Request-URI too large'
	415 -> 16#4F; %% 'Unsupported media type'
	416 -> 16#50; %% 'Requested Range Not Satisfiable'
	417 -> 16#51; %% 'Expectation Failed'
	500 -> 16#60; %% 'Internal Server Error'
	501 -> 16#61; %% 'Not Implemented'
	502 -> 16#62; %% 'Bad Gateway'
	503 -> 16#63; %% 'Service Unavailable'
	504 -> 16#64; %% 'Gateway Timeout'
	505 -> 16#65  %% 'HTTP version not supported'
    end.


decode_status_code(StatusCode) ->
    case StatusCode of
	16#10 -> 100; %% 'Continue'
	16#11 -> 101; %% 'Switching Protocols'
	16#20 -> 200; %% 'OK, Success'
	16#21 -> 201; %% 'Created'
	16#22 -> 202; %% 'Accepted'
	16#23 -> 203; %% 'Non-Authoritative Information'
	16#24 -> 204; %% 'No Content'
	16#25 -> 205; %% 'Reset Content'
	16#26 -> 206; %% 'Partial Content'
	16#30 -> 300; %% 'Multiple Choices'
	16#31 -> 301; %% 'Moved Permanently'
	16#32 -> 302; %% 'Moved temporarily'
	16#33 -> 303; %% 'See Other'
	16#34 -> 304; %% 'Not modified'
	16#35 -> 305; %% 'Use Proxy'
	16#36 -> 306; %% '(reserved)'
	16#37 -> 307; %% 'Temporary Redirect'
	16#40 -> 400; %% 'Bad Request - server could not understand request'
	16#41 -> 401; %% 'Unauthorized'
	16#42 -> 402; %% 'Payment required'
	16#43 -> 403; %% 'Forbidden operation is understood but refused'
	16#44 -> 404; %% 'Not Found'
	16#45 -> 405; %% 'Method not allowed'
	16#46 -> 406; %% 'Not Acceptable'
	16#47 -> 407; %% 'Proxy Authentication required'
	16#48 -> 408; %% 'Request Timeout'
	16#49 -> 409; %% 'Conflict'
	16#4A -> 410; %% 'Gone'
	16#4B -> 411; %% 'Length Required'
	16#4C -> 412; %% 'Precondition failed'
	16#4D -> 413; %% 'Request entity too large'
	16#4E -> 414; %% 'Request-URI too large'
	16#4F -> 415; %% 'Unsupported media type'
	16#50 -> 416; %% 'Requested Range Not Satisfiable'
	16#51 -> 417; %% 'Expectation Failed'
	16#60 -> 500; %% 'Internal Server Error'
	16#61 -> 501; %% 'Not Implemented'
	16#62 -> 502; %% 'Bad Gateway'
	16#63 -> 503; %% 'Service Unavailable'
	16#64 -> 504; %% 'Gateway Timeout'
	16#65 -> 505  %% 'HTTP version not supported'
    end.


%%
%% Content Type Assignments
%%
%% Assingment are found at http://www.wapforum.org/wina/wsp-content-type.htm
%%
%%
%% string(Version, ContentType) -> Code
%%
encode_well_known_media(ContentType, Version) ->
    case ContentType of
	%% WSP_REGISTERED_CONTENT_TYPES
	"application/vnd.uplanet.cacheop-wbxml" ->
	    encode_integer(16#0201);
	"application/vnd.uplanet.signal" ->
	    encode_integer(16#0202);
	"application/vnd.uplanet.alert-wbxml" ->
	    encode_integer(16#0203);
	"application/vnd.uplanet.list-wbxml" ->
	    encode_integer(16#0204);
	"application/vnd.uplanet.listcmd-wbxml" ->
	    encode_integer(16#0205);
	"application/vnd.uplanet.channel-wbxml" ->
	    encode_integer(16#0206);
	"application/vnd.uplanet.provisioning-status-uri" ->
	    encode_integer(16#0207);
	"x-wap.multipart/vnd.uplanet.header-set" ->
	    encode_integer(16#0208);
	"application/vnd.uplanet.bearer-choice-wbxml" ->
	    encode_integer(16#0209);
	"application/vnd.phonecom.mmc-wbxml" ->
	    encode_integer(16#020A);
	"application/vnd.nokia.syncset+wbxml" ->
	    encode_integer(16#020B);
	"image/x-up-wpng" ->
	    encode_integer(16#020C);
	_ ->
	    encode_constrained_media(ContentType, Version)
    end.


encode_constrained_media(ContentType, Version) ->
    case ContentType of
	"*/*"    -> ?ENCODE_SHORT(16#00);
	"text/*" -> ?ENCODE_SHORT(16#01);
	"text/html" -> ?ENCODE_SHORT(16#02);
	"text/plain" -> ?ENCODE_SHORT(16#03);
	"text/x-hdml" -> ?ENCODE_SHORT(16#04);
	"text/x-ttml" -> ?ENCODE_SHORT(16#05);
	"text/x-vcalendar" -> ?ENCODE_SHORT(16#06);
	"text/x-vcard" -> ?ENCODE_SHORT(16#07);
	"text/vnd.wap.wml" -> ?ENCODE_SHORT(16#08);
	"text/vnd.wap.wmlscript" -> ?ENCODE_SHORT(16#09);
	"text/vnd.wap.wta-event" -> ?ENCODE_SHORT(16#0A);
	"multipart/*" -> ?ENCODE_SHORT(16#0B);
	"multipart/mixed" -> ?ENCODE_SHORT(16#0C);
	"multipart/form-data" -> ?ENCODE_SHORT(16#0D);
	"multipart/byterantes" -> ?ENCODE_SHORT(16#0E);
	"multipart/alternative" -> ?ENCODE_SHORT(16#0F);
	"application/*" -> ?ENCODE_SHORT(16#10);
	"application/java-vm" -> ?ENCODE_SHORT(16#11);
	"application/x-www-form-urlencoded" -> ?ENCODE_SHORT(16#12);
	"application/x-hdmlc" -> ?ENCODE_SHORT(16#13);
	"application/vnd.wap.wmlc" -> ?ENCODE_SHORT(16#14);
	"application/vnd.wap.wmlscriptc" -> ?ENCODE_SHORT(16#15);
	"application/vnd.wap.wta-eventc" -> ?ENCODE_SHORT(16#16);
	"application/vnd.wap.uaprof" -> ?ENCODE_SHORT(16#17);
	"application/vnd.wap.wtls-ca-certificate" -> ?ENCODE_SHORT(16#18);
	"application/vnd.wap.wtls-user-certificate" -> ?ENCODE_SHORT(16#19);
	"application/x-x509-ca-cert" -> ?ENCODE_SHORT(16#1A);
	"application/x-x509-user-cert" -> ?ENCODE_SHORT(16#1B);
	"image/*" -> ?ENCODE_SHORT(16#1C);
	"image/gif" -> ?ENCODE_SHORT(16#1D);
	"image/jpeg" -> ?ENCODE_SHORT(16#1E);
	"image/tiff" -> ?ENCODE_SHORT(16#1F);
	"image/png" -> ?ENCODE_SHORT(16#20);
	"image/vnd.wap.wbmp" -> ?ENCODE_SHORT(16#21);
	"application/vnd.wap.multipart.*" -> ?ENCODE_SHORT(16#22);
	"application/vnd.wap.multipart.mixed" -> ?ENCODE_SHORT(16#23);
	"application/vnd.wap.multipart.form-data" -> ?ENCODE_SHORT(16#24);
	"application/vnd.wap.multipart.byteranges" -> ?ENCODE_SHORT(16#25);
	"application/vnd.wap.multipart.alternative" -> ?ENCODE_SHORT(16#26);
	"application/xml" -> ?ENCODE_SHORT(16#27);
	"text/xml" -> ?ENCODE_SHORT(16#28);
	"application/vnd.wap.wbxml" -> ?ENCODE_SHORT(16#29);
	"application/x-x968-cross-cert" -> ?ENCODE_SHORT(16#2A);
	"application/x-x968-ca-cert" -> ?ENCODE_SHORT(16#2B);
	"application/x-x968-user-cert" -> ?ENCODE_SHORT(16#2C);

	%% WAP Version 1.2
	"text/vnd.wap.si" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#2D);
	"application/vnd.wap.sic" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#2E);
	"text/vnd.wap.sl" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#2F);
	"application/vnd.wap.slc" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#30);
	"text/vnd.wap.co" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#31);
	"application/vnd.wap.coc" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#32);
	"application/vnd.wap.multipart.related" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#33);
	"application/vnd.wap.sia" when Version >= ?WSP_12 ->
	    ?ENCODE_SHORT(16#34);
	%% WAP Version 1.3
	"text/vnd.wap.connectivity-xml" when Version >= ?WSP_13 ->
	    ?ENCODE_SHORT(16#35);
	"application/vnd.wap.connectivity-wbxml" when Version >= ?WSP_13 ->
	    ?ENCODE_SHORT(16#36);
	%% WAP Version 1.4
	"application/pkcs7-mime" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#37);
	"application/vnd.wap.hashed-certificate" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#38);
	"application/vnd.wap.signed-certificate" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#39);
	"application/vnd.wap.cert-response" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3A);
	"application/xhtml+xml" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3B);
	"application/wml+xml" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3C);
	"text/css" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3D);
	"application/vnd.wap.mms-message" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3E);
	"application/vnd.wap.rollover-certificate" when Version >= ?WSP_14 ->
	    ?ENCODE_SHORT(16#3F);
	%% WAP Version 1.5
	"application/vnd.wap.locc+wbxml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#40);
	"application/vnd.wap.loc+xml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#41);
	"application/vnd.syncml.dm+wbxml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#42);
	"application/vnd.syncml.dm+xml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#43);
	"application/vnd.syncml.notification" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#44);
	"application/vnd.wap.xhtml+xml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#45);
	"application/vnd.wv.csp.cir" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#46);
	"application/vnd.oma.dd+xml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#47);
	"application/vnd.oma.drm.message" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#48);
	"application/vnd.oma.drm.content" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#49);
	"application/vnd.oma.drm.rights+xml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#4A);
	"application/vnd.oma.drm.rights+wbxml" when Version >= ?WSP_15 ->
	    ?ENCODE_SHORT(16#4B);
	_ ->
	    encode_text_string(ContentType)
    end.


decode_well_known_media(Code, Version) when integer(Code) ->
    case Code of
	%% WSP_REGISTERED_CONTENT_TYPES
	16#0201 -> "application/vnd.uplanet.cacheop-wbxml";
	16#0202 -> "application/vnd.uplanet.signal";
	16#0203 -> "application/vnd.uplanet.alert-wbxml";
	16#0204 -> "application/vnd.uplanet.list-wbxml";
	16#0205 -> "application/vnd.uplanet.listcmd-wbxml";
	16#0206 -> "application/vnd.uplanet.channel-wbxml";
	16#0207 -> "application/vnd.uplanet.provisioning-status-uri";
	16#0208 -> "x-wap.multipart/vnd.uplanet.header-set";
	16#0209 -> "application/vnd.uplanet.bearer-choice-wbxml";
	16#020A -> "application/vnd.phonecom.mmc-wbxml";
	16#020B -> "application/vnd.nokia.syncset+wbxml";
	16#020C -> "image/x-up-wpng";
	_ -> decode_constrained_media(Code, Version)
    end;
decode_well_known_media(Media, _Version) when list(Media) ->
    Media;
decode_well_known_media({short,_Data}, Version) ->
    decode_well_known_media(d_long(data), Version).	%% BUG HERE: Data


decode_constrained_media(Code, _Version) when integer(Code) ->
    case Code of
	16#00 -> "*/*";
	16#01 -> "text/*";
	16#02 -> "text/html";
	16#03 -> "text/plain";
	16#04 -> "text/x-hdml";
	16#05 -> "text/x-ttml";
	16#06 -> "text/x-vcalendar";
	16#07 -> "text/x-vcard";
	16#08 -> "text/vnd.wap.wml";
	16#09 -> "text/vnd.wap.wmlscript";
	16#0A -> "text/vnd.wap.wta-event";
	16#0B -> "multipart/*";
	16#0C -> "multipart/mixed";
	16#0D -> "multipart/form-data";
	16#0E -> "multipart/byterantes";
	16#0F -> "multipart/alternative";
	16#10 -> "application/*";
	16#11 -> "application/java-vm";
	16#12 -> "application/x-www-form-urlencoded";
	16#13 -> "application/x-hdmlc";
	16#14 -> "application/vnd.wap.wmlc";
	16#15 -> "application/vnd.wap.wmlscriptc";
	16#16 -> "application/vnd.wap.wta-eventc";
	16#17 -> "application/vnd.wap.uaprof";
	16#18 -> "application/vnd.wap.wtls-ca-certificate";
	16#19 -> "application/vnd.wap.wtls-user-certificate";
	16#1A -> "application/x-x509-ca-cert";
	16#1B -> "application/x-x509-user-cert";
	16#1C -> "image/*";
	16#1D -> "image/gif";
	16#1E -> "image/jpeg";
	16#1F -> "image/tiff";
	16#20 -> "image/png";
	16#21 -> "image/vnd.wap.wbmp";
	16#22 -> "application/vnd.wap.multipart.*";
	16#23 -> "application/vnd.wap.multipart.mixed";
	16#24 -> "application/vnd.wap.multipart.form-data";
	16#25 -> "application/vnd.wap.multipart.byteranges";
	16#26 -> "application/vnd.wap.multipart.alternative";
	16#27 -> "application/xml";
	16#28 -> "text/xml";
	16#29 -> "application/vnd.wap.wbxml";
	16#2A -> "application/x-x968-cross-cert";
	16#2B -> "application/x-x968-ca-cert";
	16#2C -> "application/x-x968-user-cert";
	%% WAP Version 1.2
	16#2D -> "text/vnd.wap.si";
	16#2E -> "application/vnd.wap.sic";
	16#2F -> "text/vnd.wap.sl";
	16#30 -> "application/vnd.wap.slc";
	16#31 -> "text/vnd.wap.co";
	16#32 -> "application/vnd.wap.coc";
	16#33 -> "application/vnd.wap.multipart.related";
	16#34 -> "application/vnd.wap.sia";
	%% WAP Version 1.3
	16#35 -> "text/vnd.wap.connectivity-xml";
	16#36 -> "application/vnd.wap.connectivity-wbxml";
	%% WAP Version 1.4
	16#37 -> "application/pkcs7-mime";
	16#38 -> "application/vnd.wap.hashed-certificate";
	16#39 -> "application/vnd.wap.signed-certificate";
	16#3A -> "application/vnd.wap.cert-response";
	16#3B -> "application/xhtml+xml";
	16#3C -> "application/wml+xml";
	16#3D -> "text/css";
	16#3E -> "application/vnd.wap.mms-message";
	16#3F -> "application/vnd.wap.rollover-certificate";
	%% WAP Version 1.5
	16#40 -> "application/vnd.wap.locc+wbxml";
	16#41 -> "application/vnd.wap.loc+xml";
	16#42 -> "application/vnd.syncml.dm+wbxml";
	16#43 -> "application/vnd.syncml.dm+xml";
	16#44 -> "application/vnd.syncml.notification";
	16#45 -> "application/vnd.wap.xhtml+xml";
	16#46 -> "application/vnd.wv.csp.cir";
	16#47 -> "application/vnd.oma.dd+xml";
	16#48 -> "application/vnd.oma.drm.message";
	16#49 -> "application/vnd.oma.drm.content";
	16#4A -> "application/vnd.oma.drm.rights+xml";
	16#4B -> "application/vnd.oma.drm.rights+wbxml"
    end;
decode_constrained_media(Media, _Version) when list(Media) ->
    Media.


%% Parse <integer> or <integer>.<integer>

parse_version(Value) ->
    case string:tokens(Value, ".") of
	[Major,Minor] ->
	    {list_to_integer(Major), list_to_integer(Minor)};
	[Major] ->
	    case catch list_to_integer(Major) of
		{'EXIT', _} ->
		    Value;
		V -> V
	    end
    end.

format_version({Major,Minor}) ->
    [integer_to_list(Major),".",integer_to_list(Minor)];
format_version(Major) when integer(Major) ->
    integer_to_list(Major);
format_version(Version) when list(Version) ->
    Version.

encode_version({Major,Minor}) ->
    Ver = (((Major-1) band 16#7) bsl 4) bor (Minor band 16#f),
    ?ENCODE_SHORT(Ver);
encode_version(Major) when integer(Major) ->
    Ver = ((Major band 16#7) bsl 4) bor 16#f,
    ?ENCODE_SHORT(Ver);
encode_version(Value) when list(Value) ->
    encode_text_string(Value).


decode_version(Value) when integer(Value) ->
    Major = (Value bsr 4) band 16#7,
    Minor = Value band 16#f,
    if Minor == 16#f ->
	    Major;
       true ->
	    {Major+1,Minor}
    end;
decode_version(Value) when list(Value) ->
    Value.


encode_mms_version({Major,Minor}) ->
    Ver = ((Major band 16#7) bsl 4) bor (Minor band 16#f),
    ?ENCODE_SHORT(Ver);
encode_mms_version(Major) when integer(Major) ->
    Ver = ((Major band 16#7) bsl 4) bor 16#f,
    ?ENCODE_SHORT(Ver);
encode_mms_version(Value) when list(Value) ->
    encode_text_string(Value).


decode_mms_version(Value) when integer(Value) ->
    Major = (Value bsr 4) band 16#7,
    Minor = Value band 16#f,
    if Minor == 16#f ->
	    Major;
       true ->
	    {Major,Minor}
    end;
decode_mms_version(Value) when list(Value) ->
    Value.


%%%
%%% Basic data types
%%%

e_delta_seconds(Value) ->
    encode_integer(Value).


encode_integer(I) when integer(I), I >= 0 , I < 127 ->
    ?ENCODE_SHORT(I);
encode_integer(I) when integer(I) ->
    encode_long_integer(I);
encode_integer(List) when list(List) ->
    encode_integer(list_to_integer(List)).

decode_integer(Value) when integer(Value) ->
    Value;
decode_integer({short,Data}) ->
    Sz = size(Data)*8,
    <<Value:Sz>> = Data,
    Value.

encode_short_integer(I) ->
    ?ENCODE_SHORT(I).

encode_long_integer(I) when I >= 0 ->
    MOInt = encode_multioctet_integer(I, []),
    MOIntLen = length(MOInt),
    list_to_binary([MOIntLen band 16#1f | MOInt]).

encode_multioctet_integer(I,Acc) when I < 256 ->
    [I | Acc];
encode_multioctet_integer(I,Acc) ->
    encode_multioctet_integer(I bsr 8, [(I band 16#ff) | Acc]).


%% Integer-Value: Short-Integer | Long-Integer
%% Short-Integer: <<1:Short:7>>
%% Long-Integer:  <<0-30, X:0-30>>
%% return {Integer,Tail}
d_integer_value(<<1:1,Integer:7,Tail/binary>>) ->
    {Integer, Tail};
d_integer_value(<<0:3,Len:5,Data/binary>>) when Len =/= 31 ->
    Sz = Len*8,
    <<Integer:Sz, Tail/binary>> = Data,
    {Integer, Tail}.

decode_short_integer(<<1:1,Septet:7,T100/binary>>) ->
    {Septet, T100}.

decode_long_integer(<<0:3,Len:5,Data/binary>>) when Len =/= 31 ->
    Sz = Len*8,
    <<Val:Sz, Tail/binary>> = Data,
    {Val, Tail}.

d_long(Data) ->
    Sz = size(Data)*8,
    <<Value:Sz>> = Data,
    Value.


encode_uri_value(Data) ->
    encode_text_string(Data).

decode_uri_value(Data) when list(Data) ->
    Data.

%% parse quoted string
decode_quoted_string([$" | List]) ->
    List.

encode_quoted_string([$" | Value]) ->
    case lists:reverse(Value) of
	[$" | Value1] ->
	    <<$", (list_to_binary(lists:reverse(Value1)))/binary, 0>>;
	_ ->
	    <<$", (list_to_binary(Value))/binary, 0>>
    end;
encode_quoted_string(Value) ->
    <<$", (list_to_binary(Value))/binary, 0>>.



decode_text_string(List) when list(List) ->
    List;
decode_text_string(Bin) when binary(Bin) ->
    binary_to_list(Bin).



encode_text_string(A) when atom(A) ->
    encode_text_string(atom_to_list(A));
encode_text_string([H|T]) when H >= 128 ->
    <<(list_to_binary([127,H|T]))/binary,0>>;
encode_text_string(S) ->
    <<(list_to_binary(S))/binary,0>>.


encode_text_value(undefined) ->
    <<0>>;
encode_text_value([$"|T]) ->
    %% remove ending quote ?
    <<34,(list_to_binary(T))/binary>>;
encode_text_value(L) ->
    encode_text_string(L).


d_text_value(<<0,T100/binary>>) ->
    { "", T100};
d_text_value(<<34,_Tail/binary>>=Data) ->
    d_text_string(Data);
d_text_value(Data) ->
    d_text_string(Data).


d_text_string(<<127,Data/binary>>) -> %% Remove quote
    d_text_string(Data,[]);
d_text_string(Data) ->
    d_text_string(Data,[]).

d_text_string(<<0,Tail/binary>>,A) ->
    {lists:reverse(A), Tail};
d_text_string(<<C,Tail/binary>>,A) ->
    d_text_string(Tail,[C|A]);
d_text_string(<<>>, A) ->
    {lists:reverse(A), <<>>}.


d_q_value(<<0:1,Q:7,Tail/binary>>) ->
    QVal =
	if Q >= 1, Q =< 100 ->
		lists:flatten(io_lib:format("0.~2..0w", [Q-1]));
	   Q >= 101, Q =< 1099 ->
		lists:flatten(io_lib:format("0.~3..0w", [Q-100]));
	   true ->
		io:format("Q-value to big ~w\n", [Q]),
		"***"
	end,
    {QVal, Tail};
d_q_value(<<1:1,Q1:7,0:1,Q0:7,Tail/binary>>) ->
    Q = (Q1 bsl 7) bor Q0,
    QVal =
	if Q >= 1, Q =< 100 ->
		lists:flatten(io_lib:format("0.~2..0w", [Q-1]));
	   Q >= 101, Q =< 1099 ->
		lists:flatten(io_lib:format("0.~3..0w", [Q-100]));
	   true ->
		io:format("Q-value to big ~w\n", [Q]),
		"***"
	end,
    {QVal, Tail}.


%%
%% Decode uintvar
%%
d_uintvar(<<0:1,S0:7,T100/binary>>) ->
    {S0, T100};
d_uintvar(<<1:1,S1:7,0:1,S0:7,T100/binary>>) ->
    {(S1 bsl 7) bor S0, T100};
d_uintvar(<<1:1,S2:7,1:1,S1:7,0:1,S0:7,T100/binary>>) ->
    {(S2 bsl 14) bor (S1 bsl 7) bor S0, T100};
d_uintvar(<<1:1,S3:7,1:1,S2:7,1:1,S1:7,0:1,S0:7,T100/binary>>) ->
    {(S3 bsl 21) bor (S2 bsl 14) bor (S1 bsl 7) bor S0, T100};
d_uintvar(<<1:1,S4:7,1:1,S3:7,1:1,S2:7,1:1,S1:7,0:1,S0:7,T100/binary>>) ->
    {(S4 bsl 28) bor (S3 bsl 21) bor (S2 bsl 14) bor (S1 bsl 7) bor S0, T100}.


e_uintvar(I) when I < 128 -> <<I>>;
e_uintvar(I) -> e_uintvar(I,[]).

e_uintvar(0,Acc) ->
    list_to_binary(Acc);
e_uintvar(I,[]) ->
    e_uintvar(I bsr 7, [I band 16#7f]);
e_uintvar(I,Acc) ->
    e_uintvar(I bsr 7, [16#80 bor (I band 16#7f) | Acc]).


e_value(B) ->
    Sz = size(B),
    if Sz =< 30 ->
	    <<Sz:8, B/binary>>;
       true ->
	    <<31:8, (e_uintvar(Sz))/binary, B/binary >>
    end.

e_value(B1,B2) ->
    Sz = size(B1)+size(B2),
    if Sz =< 30 ->
	    <<Sz:8, B1/binary, B2/binary>>;
       true ->
	    <<31:8, (e_uintvar(Sz))/binary, B1/binary, B2/binary >>
    end.

e_value(B1,B2,B3) ->
    Sz = size(B1)+size(B2)+size(B3),
    if Sz =< 30 ->
	    <<Sz:8, B1/binary,B2/binary,B3/binary>>;
       true ->
	    <<31:8,(e_uintvar(Sz))/binary,B1/binary,B2/binary,B3/binary>>
    end.

e_value(B1,B2,B3,B4) ->
    Sz = size(B1)+size(B2)+size(B3)+size(B4),
    if Sz =< 30 ->
	    <<Sz:8, B1/binary,B2/binary,B3/binary,B4/binary>>;
       true ->
	    <<31:8,(e_uintvar(Sz))/binary,B1/binary,
	     B2/binary,B3/binary,B4/binary>>
    end.

%%
%% Extened methods
%%
decode_extended_methods(<<PduType:8, Data/binary>>) ->
    Type = decode_pdu_type(PduType),
    {Method, Data1} = d_text_string(Data),
    [{Type,Method} | decode_extended_methods(Data1)];
decode_extended_methods(<<>>) ->
    [].

encode_extended_methods(Ms) ->
    list_to_binary(encode_ext_methods(Ms)).

encode_ext_methods([{Type,Method} | T]) ->
    [ encode_pdu_type(Type), encode_text_string(Method) |
      encode_ext_methods(T)];
encode_ext_methods([]) ->
    [].

%%
%% Address lists used by redirect-pdu and aliases-capability
%%
decode_address(D0) ->
    [A] = decode_addresses(D0),
    A.

decode_addresses(D0) ->
    case D0 of
	<<1:1, 1:1,Len:6,B:8,P:16,Addr:Len/binary,D1/binary>> ->
	    [#wdp_address { bearer = B, address = Addr, portnum=P } |
	     decode_addresses(D1)];
	<<1:1, 0:1,Len:6,B:8,Addr:Len/binary,D1/binary>> ->
	    [#wdp_address { bearer = B, address = Addr } |
	     decode_addresses(D1)];
	<<0:1, 1:1,Len:6,P:16,Addr:Len/binary,D1/binary>> ->
	    [#wdp_address { portnum=P, address=Addr } |
	     decode_addresses(D1)];
	<<0:1, 0:1,Len:6,Addr:Len/binary,D1/binary>> ->
	    [#wdp_address { address=Addr } |
	     decode_addresses(D1)];
	<<>> ->
	    []
    end.

encode_addresses(As) ->
    encode_addresses(As, []).

encode_addresses([A|As], Acc) ->
    encode_addresses(As, [encode_address(A)|Acc]);
encode_addresses([], Acc) ->
    list_to_binary(lists:reverse(Acc)).

encode_address(#wdp_address { bearer = B, address = Addr, portnum = P }) ->
    BAddr = if tuple(Addr) ->
		    list_to_binary(inet:ip_to_bytes(Addr));
	       binary(Addr) ->
		    Addr
	    end,
    Len = size(BAddr),
    if B == undefined, P == undefined ->
	    <<0:1, 0:1, Len:6, BAddr/binary>>;
       B == undefined ->
	    <<0:1, 1:1, Len:6, P:16, BAddr/binary>>;
       P == undefined ->
	    <<1:1, 0:1, Len:6, B:8, BAddr/binary>>;
       true ->
	    <<1:1, 1:1, Len:6, B:8, P:16, BAddr/binary>>
    end.




-define(UNIX_TIME_OFFSET, 62167219200).

d_date(Val) when integer(Val) ->
    calendar:gregorian_seconds_to_datetime(Val+?UNIX_TIME_OFFSET);
d_date({short,Data}) ->
    Sz = size(Data)*8,
    <<Sec:Sz>> = Data,
    calendar:gregorian_seconds_to_datetime(Sec+?UNIX_TIME_OFFSET).

e_date(DateTime) ->
    Sec = calendar:datetime_to_gregorian_seconds(DateTime),
    encode_long_integer(Sec -  ?UNIX_TIME_OFFSET).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% decode http-date (RFC 2068). (MUST be send in RFC1123 date format)
%%          HTTP-date    = rfc1123-date | rfc850-date | asctime-date
%%          rfc1123-date = wkday "," SP date1 SP time SP "GMT"
%%          rfc850-date  = weekday "," SP date2 SP time SP "GMT"
%%          asctime-date = wkday SP date3 SP time SP 4DIGIT
%%
%%          date1        = 2DIGIT SP month SP 4DIGIT
%%                         ; day month year (e.g., 02 Jun 1982)
%%          date2        = 2DIGIT "-" month "-" 2DIGIT
%%                         ; day-month-year (e.g., 02-Jun-82)
%%          date3        = month SP ( 2DIGIT | ( SP 1DIGIT ))
%%                         ; month day (e.g., Jun  2)
%%
%%          time         = 2DIGIT ":" 2DIGIT ":" 2DIGIT
%%                         ; 00:00:00 - 23:59:59
%%
%%          wkday        = "Mon" | "Tue" | "Wed"
%%                       | "Thu" | "Fri" | "Sat" | "Sun"
%%
%%
%%          weekday      = "Monday" | "Tuesday" | "Wednesday"
%%                       | "Thursday" | "Friday" | "Saturday" | "Sunday"
%%
%%          month        = "Jan" | "Feb" | "Mar" | "Apr"
%%                       | "May" | "Jun" | "Jul" | "Aug"
%%                       | "Sep" | "Oct" | "Nov" | "Dec"
%%
%% decode date or crash!
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
parse_http_date(Date) ->
    parse_hdate(tolower(Date)).

parse_hdate([$m,$o,$n,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$t,$u,$e,$s,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$w,$e,$d,$n,$s,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$t,$h,$u,$r,$s,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$f,$r,$i,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$s,$a,$t,$u,$r,$d,$a,$y,$   | Cs]) -> date2(Cs);
parse_hdate([$s,$u,$n,$d,$a,$y,$  | Cs]) -> date2(Cs);
parse_hdate([$m,$o,$n,X | Cs]) -> date13(X,Cs);
parse_hdate([$t,$u,$e,X  | Cs]) -> date13(X,Cs);
parse_hdate([$w,$e,$d,X  | Cs]) -> date13(X,Cs);
parse_hdate([$t,$h,$u,X  | Cs]) -> date13(X,Cs);
parse_hdate([$f,$r,$i,X  | Cs]) -> date13(X,Cs);
parse_hdate([$s,$a,$t,X  | Cs]) -> date13(X,Cs);
parse_hdate([$s,$u,$n,X  | Cs]) -> date13(X,Cs).

date13($ , Cs) -> date3(Cs);
date13($,, [$ |Cs]) -> date1(Cs).

%% date1
date1([D1,D2,$ ,M1,M2,M3,$ ,Y1,Y2,Y3,Y4,$  | Cs]) ->
    M = parse_month([M1,M2,M3]),
    D = list_to_integer([D1,D2]),
    Y = list_to_integer([Y1,Y2,Y3,Y4]),
    {Time,[$ ,$g,$m,$t|Cs1]} = parse_time(Cs),
    { {{Y,M,D},Time}, Cs1}.

%% date2
date2([D1,D2,$-,M1,M2,M3,$-,Y1,Y2 | Cs]) ->
    M = parse_month([M1,M2,M3]),
    D = list_to_integer([D1,D2]),
    Y = 1900 + list_to_integer([Y1,Y2]),
    {Time, [$ ,$g,$m,$t|Cs1]} = parse_time(Cs),
    {{{Y,M,D}, Time}, Cs1}.

%% date3
date3([M1,M2,M3,$ ,D1,D2,$ | Cs]) ->
    M = parse_month([M1,M2,M3]),
    D = if D1 == $  -> list_to_integer([D2]);
	   true -> list_to_integer([D1,D2])
	end,
    {Time,[$ ,Y1,Y2,Y3,Y4|Cs1]} = parse_time(Cs),
    Y = list_to_integer([Y1,Y2,Y3,Y4]),
    { {{Y,M,D}, Time}, Cs1 }.

%% decode lowercase month
parse_month("jan") -> 1;
parse_month("feb") -> 2;
parse_month("mar") -> 3;
parse_month("apr") -> 4;
parse_month("may") -> 5;
parse_month("jun") -> 6;
parse_month("jul") -> 7;
parse_month("aug") -> 8;
parse_month("sep") -> 9;
parse_month("oct") -> 10;
parse_month("nov") -> 11;
parse_month("dec") -> 12.

%% decode time HH:MM:SS
parse_time([H1,H2,$:,M1,M2,$:,S1,S2|Cs]) ->
    { {list_to_integer([H1,H2]),
       list_to_integer([M1,M2]),
       list_to_integer([S1,S2]) }, Cs}.

%% encode date into rfc1123-date (must be a GMT time!!!)
fmt_date({{Y,M,D},{TH,TM,TS}}) ->
    WkDay = case calendar:day_of_the_week({Y,M,D}) of
		1 -> "Mon";
		2 -> "Tue";
		3 -> "Wed";
		4 -> "Thu";
		5 -> "Fri";
		6 -> "Sat";
		7 -> "Sun"
	    end,
    lists:flatten(io_lib:format("~s, ~2..0w ~s ~4..0w "
				"~2..0w:~2..0w:~2..0w GMT",
				[WkDay, D, fmt_month(M), Y, TH, TM, TS])).

fmt_current_date() ->
    fmt_date(calendar:universal_time()).

%% decode lowercase month
fmt_month(1) -> "Jan";
fmt_month(2) -> "Feb";
fmt_month(3) -> "Mar";
fmt_month(4) -> "Apr";
fmt_month(5) -> "May";
fmt_month(6) -> "Jun";
fmt_month(7) -> "Jul";
fmt_month(8) -> "Aug";
fmt_month(9) -> "Sep";
fmt_month(10) -> "Oct";
fmt_month(11) -> "Nov";
fmt_month(12) -> "Dec".